LyoKICoJaGFuZGxpbmcgb2YgU0hFTEwzMi5ETEwgT0xFLU9iamVjdHMKICoKICoJQ29weXJpZ2h0IDE5OTcJTWFyY3VzIE1laXNzbmVyCiAqCUNvcHlyaWdodCAxOTk4CUp1ZXJnZW4gU2NobWllZCAgPGp1ZXJnZW4uc2NobWllZEBtZXRyb25ldC5kZT4KICoKICovCgojaW5jbHVkZSA8c3RkbGliLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KCiNpbmNsdWRlICJ3aW5lL29ial9iYXNlLmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGxpbmsuaCIKI2luY2x1ZGUgIndpbmUvb2JqX3NoZWxsZm9sZGVyLmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGJyb3dzZXIuaCIKI2luY2x1ZGUgIndpbmUvb2JqX2NvbnRleHRtZW51LmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGV4dGluaXQuaCIKI2luY2x1ZGUgIndpbmUvb2JqX2V4dHJhY3RpY29uLmgiCgojaW5jbHVkZSAic2hsZ3VpZC5oIgojaW5jbHVkZSAid2ludmVyc2lvbi5oIgojaW5jbHVkZSAid2lucmVnLmgiCiNpbmNsdWRlICJ3aW5lcnJvci5oIgojaW5jbHVkZSAiZGVidWd0b29scy5oIgoKI2luY2x1ZGUgInNoZWxsMzJfbWFpbi5oIgoKREVGQVVMVF9ERUJVR19DSEFOTkVMKHNoZWxsKQoKRFdPUkQgV0lOQVBJIFNIQ0xTSURGcm9tU3RyaW5nQSAoTFBTVFIgY2xzaWQsIENMU0lEICppZCk7CmV4dGVybiBJU2hlbGxGb2xkZXIgKiBJU2hlbGxGb2xkZXJfQ29uc3RydWN0b3IoCglJU2hlbGxGb2xkZXIgKiBwc2YsCglMUElURU1JRExJU1QgcGlkbCk7CmV4dGVybiBIUkVTVUxUIElGU0ZvbGRlcl9Db25zdHJ1Y3RvcigKCUlVbmtub3duICogcFVua091dGVyLAoJUkVGSUlEIHJpaWQsCglMUFZPSUQgKiBwcHYpOwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogU0hDb0NyZWF0ZUluc3RhbmNlIFtTSEVMTDMyLjEwMl0KICogCiAqIE5PVEVTCiAqICAgICBleHBvcnRlZCBieSBvcmRpbmFsCiAqLwpMUkVTVUxUIFdJTkFQSSBTSENvQ3JlYXRlSW5zdGFuY2UoCglMUFNUUiBhY2xzaWQsCglSRUZDTFNJRCBjbHNpZCwKCUlVbmtub3duICogdW5rbm93bm91dGVyLAoJUkVGSUlEIHJlZmlpZCwKCUxQVk9JRCAqcHB2KQp7CglEV09SRAlocmVzOwoJSUlECWlpZDsKCUNMU0lEICogbXljbHNpZCA9IChDTFNJRCopY2xzaWQ7CgkKCWlmICghY2xzaWQpCgl7CgkgIGlmICghYWNsc2lkKSByZXR1cm4gUkVHREJfRV9DTEFTU05PVFJFRzsKCSAgU0hDTFNJREZyb21TdHJpbmdBKGFjbHNpZCwgJmlpZCk7CgkgIG15Y2xzaWQgPSAmaWlkOwoJfQoKCVRSQUNFKCIoJXAsXG5cdENMU0lEOlx0JXMsIHVuazolcFxuXHRJSUQ6XHQlcywlcClcbiIsCgkJYWNsc2lkLGRlYnVnc3RyX2d1aWQobXljbHNpZCksdW5rbm93bm91dGVyLGRlYnVnc3RyX2d1aWQocmVmaWlkKSxwcHYpOwoKCWlmIElzRXF1YWxDTFNJRChteWNsc2lkLCAmQ0xTSURfU2hlbGxGU0ZvbGRlcikKCXsKCSAgaHJlcyA9IElGU0ZvbGRlcl9Db25zdHJ1Y3Rvcih1bmtub3dub3V0ZXIsIHJlZmlpZCwgcHB2KTsKCX0KCWVsc2UKCXsKCSAgaHJlcyA9IENvQ3JlYXRlSW5zdGFuY2UobXljbHNpZCwgdW5rbm93bm91dGVyLCBDTFNDVFhfSU5QUk9DX1NFUlZFUiwgcmVmaWlkLCBwcHYpOwoJfQoJCglpZihocmVzIT1TX09LKQoJewoJICBFUlIoImZhaWxlZCAoMHglMDhseCkgdG8gY3JlYXRlIFxuXHRDTFNJRDpcdCVzXG5cdElJRDpcdCVzXG4iLAogICAgICAgICAgICAgIGhyZXMsIGRlYnVnc3RyX2d1aWQobXljbHNpZCksIGRlYnVnc3RyX2d1aWQocmVmaWlkKSk7CgkgIEVSUigiY2xhc3Mgbm90IGZvdW5kIGluIHJlZ2lzdHJ5XG4iKTsKCX0KCglUUkFDRSgiLS0gaW5zdGFuY2U6ICVwXG4iLCpwcHYpOwoJcmV0dXJuIGhyZXM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIFNIRUxMMzJfRGxsR2V0Q2xhc3NPYmplY3QgICBbU0hFTEwzMi4xMjhdCiAqLwpIUkVTVUxUIFdJTkFQSSBTSEVMTDMyX0RsbEdldENsYXNzT2JqZWN0KFJFRkNMU0lEIHJjbHNpZCwgUkVGSUlEIGlpZCxMUFZPSUQgKnBwdikKewlIUkVTVUxUCWhyZXMgPSBFX09VVE9GTUVNT1JZOwoJTFBDTEFTU0ZBQ1RPUlkgbHBjbGY7CgoJVFJBQ0UoIlxuXHRDTFNJRDpcdCVzLFxuXHRJSUQ6XHQlc1xuIixkZWJ1Z3N0cl9ndWlkKHJjbHNpZCksZGVidWdzdHJfZ3VpZChpaWQpKTsKCQoJKnBwdiA9IE5VTEw7CgoJaWYoSXNFcXVhbENMU0lEKHJjbHNpZCwgJkNMU0lEX1NoZWxsRGVza3RvcCl8fCAKCSAgIElzRXF1YWxDTFNJRChyY2xzaWQsICZDTFNJRF9TaGVsbExpbmspKQoJewoJICBscGNsZiA9IElDbGFzc0ZhY3RvcnlfQ29uc3RydWN0b3IoIHJjbHNpZCApOwoKCSAgaWYobHBjbGYpIAoJICB7CgkgICAgaHJlcyA9IElDbGFzc0ZhY3RvcnlfUXVlcnlJbnRlcmZhY2UobHBjbGYsaWlkLCBwcHYpOwoJICAgIElDbGFzc0ZhY3RvcnlfUmVsZWFzZShscGNsZik7CgkgIH0KCX0KCWVsc2UKCXsKCSAgV0FSTigiLS0gQ0xTSUQgbm90IGZvdW5kXG4iKTsKCSAgaHJlcyA9IENMQVNTX0VfQ0xBU1NOT1RBVkFJTEFCTEU7Cgl9CglUUkFDRSgiLS0gcG9pbnRlciB0byBjbGFzcyBmYWN0b3J5OiAlcFxuIiwqcHB2KTsKCXJldHVybiBocmVzOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBTSENMU0lERnJvbVN0cmluZwkJCQlbU0hFTEwzMi4xNDddCiAqCiAqIE5PVEVTCiAqICAgICBleHBvcnRlZCBieSBvcmRpbmFsCiAqLwpEV09SRCBXSU5BUEkgU0hDTFNJREZyb21TdHJpbmdBIChMUFNUUiBjbHNpZCwgQ0xTSUQgKmlkKQp7CglUUkFDRSgiKCVwKCVzKSAlcClcbiIsIGNsc2lkLCBjbHNpZCwgaWQpOwoJcmV0dXJuIENMU0lERnJvbVN0cmluZzE2KGNsc2lkLCBpZCk7IAp9CkRXT1JEIFdJTkFQSSBTSENMU0lERnJvbVN0cmluZ1cgKExQV1NUUiBjbHNpZCwgQ0xTSUQgKmlkKQp7CglUUkFDRSgiKCVwKCVzKSAlcClcbiIsIGNsc2lkLCBkZWJ1Z3N0cl93KGNsc2lkKSwgaWQpOwoJcmV0dXJuIENMU0lERnJvbVN0cmluZyhjbHNpZCwgaWQpOyAKfQpEV09SRCBXSU5BUEkgU0hDTFNJREZyb21TdHJpbmdBVyAoTFBWT0lEIGNsc2lkLCBDTFNJRCAqaWQpCnsKCWlmIChWRVJTSU9OX09zSXNVbmljb2RlKCkpCgkgIHJldHVybiBTSENMU0lERnJvbVN0cmluZ1cgKGNsc2lkLCBpZCk7CglyZXR1cm4gU0hDTFNJREZyb21TdHJpbmdBIChjbHNpZCwgaWQpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJCSBTSEdldE1hbGxvYwkJCVtTSEVMTDMyLjIyMF0KICogcmV0dXJucyB0aGUgaW50ZXJmYWNlIHRvIHNoZWxsIG1hbGxvYy4KICoKICogW1NESyBoZWFkZXIgd2luOTUvc2hsb2JqLmg6CiAqIGVxdWl2YWxlbnQgdG86ICAjZGVmaW5lIFNIR2V0TWFsbG9jKHBwbWVtKSAgIENvR2V0TWFsbG9jKE1FTUNUWF9UQVNLLCBwcG1lbSkKICogXQogKiBXaGF0IHdlIGFyZSBjdXJyZW50bHkgZG9pbmcgaXMgbm90IHZlcnkgd3JvbmcsIHNpbmNlIHdlIGFsd2F5cyB1c2UgdGhlIHNhbWUKICogaGVhcCAoUHJvY2Vzc0hlYXApLgogKi8KRFdPUkQgV0lOQVBJIFNIR2V0TWFsbG9jKExQTUFMTE9DICpscG1hbCkgCnsKCVRSQUNFKCIoJXApXG4iLCBscG1hbCk7CglyZXR1cm4gQ29HZXRNYWxsb2MoMCxscG1hbCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIFNIR2V0RGVza3RvcEZvbGRlcgkJCVtTSEVMTDMyLjIxNl0KICovCkxQU0hFTExGT0xERVIgcGRlc2t0b3Bmb2xkZXI9TlVMTDsKCkRXT1JEIFdJTkFQSSBTSEdldERlc2t0b3BGb2xkZXIoSVNoZWxsRm9sZGVyICoqcHNmKQp7CglIUkVTVUxUCWhyZXMgPSBTX09LOwoJTFBDTEFTU0ZBQ1RPUlkgbHBjbGY7CglUUkFDRSgiJXAtPiglcClcbiIscHNmLCpwc2YpOwoKCSpwc2Y9TlVMTDsKCglpZiAoIXBkZXNrdG9wZm9sZGVyKSAKCXsKCSAgbHBjbGYgPSBJQ2xhc3NGYWN0b3J5X0NvbnN0cnVjdG9yKCZDTFNJRF9TaGVsbERlc2t0b3ApOwoJICBpZihscGNsZikgCgkgIHsKCSAgICBocmVzID0gSUNsYXNzRmFjdG9yeV9DcmVhdGVJbnN0YW5jZShscGNsZixOVUxMLChSRUZJSUQpJklJRF9JU2hlbGxGb2xkZXIsICh2b2lkKikmcGRlc2t0b3Bmb2xkZXIpOwoJICAgIElDbGFzc0ZhY3RvcnlfUmVsZWFzZShscGNsZik7CgkgIH0gIAoJfQoJCglpZiAocGRlc2t0b3Bmb2xkZXIpIAoJewoJICAvKiBldmVuIGlmIHdlIGNyZWF0ZSB0aGUgZm9sZGVyLCBhZGQgYSByZWYgc28gdGhlIGFwcGxpY2F0aW9uIGNhbrR0IGRlc3Ryb3kgdGhlIGZvbGRlciovCgkgIElTaGVsbEZvbGRlcl9BZGRSZWYocGRlc2t0b3Bmb2xkZXIpOwoJICAqcHNmID0gcGRlc2t0b3Bmb2xkZXI7Cgl9CgoJVFJBQ0UoIi0tICVwLT4oJXApXG4iLHBzZiwgKnBzZik7CglyZXR1cm4gaHJlczsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiogIElDbGFzc0ZhY3RvcnkgSW1wbGVtZW50YXRpb24KKi8KCnR5cGVkZWYgc3RydWN0CnsKICAgIC8qIElVbmtub3duIGZpZWxkcyAqLwogICAgSUNPTV9WRklFTEQoSUNsYXNzRmFjdG9yeSk7CiAgICBEV09SRCAgICAgICAgICAgICAgICAgICAgICAgcmVmOwogICAgQ0xTSUQJCQkqcmNsc2lkOwp9IElDbGFzc0ZhY3RvcnlJbXBsOwoKc3RhdGljIElDT01fVlRBQkxFKElDbGFzc0ZhY3RvcnkpIGNsZnZ0OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICBJQ2xhc3NGYWN0b3J5X0NvbnN0cnVjdG9yCiAqLwoKTFBDTEFTU0ZBQ1RPUlkgSUNsYXNzRmFjdG9yeV9Db25zdHJ1Y3RvcihSRUZDTFNJRCByY2xzaWQpCnsKCUlDbGFzc0ZhY3RvcnlJbXBsKiBscGNsZjsKCglscGNsZj0gKElDbGFzc0ZhY3RvcnlJbXBsKilIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwwLHNpemVvZihJQ2xhc3NGYWN0b3J5SW1wbCkpOwoJbHBjbGYtPnJlZiA9IDE7CglJQ09NX1ZUQkwobHBjbGYpID0gJmNsZnZ0OwoJbHBjbGYtPnJjbHNpZCA9IChDTFNJRCopcmNsc2lkOwoKCVRSQUNFKCIoJXApLT4oKVxuIixscGNsZik7CglJbnRlcmxvY2tlZEluY3JlbWVudCgmc2hlbGwzMl9PYmpDb3VudCk7CglyZXR1cm4gKExQQ0xBU1NGQUNUT1JZKWxwY2xmOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgSUNsYXNzRmFjdG9yeV9RdWVyeUludGVyZmFjZQogKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJIElDbGFzc0ZhY3RvcnlfZm5RdWVyeUludGVyZmFjZSgKICBMUENMQVNTRkFDVE9SWSBpZmFjZSwgUkVGSUlEIHJpaWQsIExQVk9JRCAqcHB2T2JqKQp7CglJQ09NX1RISVMoSUNsYXNzRmFjdG9yeUltcGwsaWZhY2UpOwoJVFJBQ0UoIiglcCktPihcblx0SUlEOlx0JXMpXG4iLFRoaXMsZGVidWdzdHJfZ3VpZChyaWlkKSk7CgoJKnBwdk9iaiA9IE5VTEw7CgoJaWYoSXNFcXVhbElJRChyaWlkLCAmSUlEX0lVbmtub3duKSkgICAgICAgICAgLypJVW5rbm93biovCgl7ICpwcHZPYmogPSBUaGlzOyAKCX0KCWVsc2UgaWYoSXNFcXVhbElJRChyaWlkLCAmSUlEX0lDbGFzc0ZhY3RvcnkpKSAgLypJQ2xhc3NGYWN0b3J5Ki8KCXsgKnBwdk9iaiA9IChJQ2xhc3NGYWN0b3J5KilUaGlzOwoJfSAgIAoKCWlmKCpwcHZPYmopCgl7IElVbmtub3duX0FkZFJlZigoTFBVTktOT1dOKSpwcHZPYmopOyAgCQoJICBUUkFDRSgiLS0gSW50ZXJmYWNlOiAoJXApLT4oJXApXG4iLHBwdk9iaiwqcHB2T2JqKTsKCSAgcmV0dXJuIFNfT0s7Cgl9CglUUkFDRSgiLS0gSW50ZXJmYWNlOiAlcyBFX05PSU5URVJGQUNFXG4iLCBkZWJ1Z3N0cl9ndWlkKHJpaWQpKTsKCXJldHVybiBFX05PSU5URVJGQUNFOwp9ICAKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJQ2xhc3NGYWN0b3J5X0FkZFJlZgogKi8Kc3RhdGljIFVMT05HIFdJTkFQSSBJQ2xhc3NGYWN0b3J5X2ZuQWRkUmVmKExQQ0xBU1NGQUNUT1JZIGlmYWNlKQp7CglJQ09NX1RISVMoSUNsYXNzRmFjdG9yeUltcGwsaWZhY2UpOwoJVFJBQ0UoIiglcCktPihjb3VudD0lbHUpXG4iLFRoaXMsVGhpcy0+cmVmKTsKCglJbnRlcmxvY2tlZEluY3JlbWVudCgmc2hlbGwzMl9PYmpDb3VudCk7CglyZXR1cm4gSW50ZXJsb2NrZWRJbmNyZW1lbnQoJlRoaXMtPnJlZik7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJQ2xhc3NGYWN0b3J5X1JlbGVhc2UKICovCnN0YXRpYyBVTE9ORyBXSU5BUEkgSUNsYXNzRmFjdG9yeV9mblJlbGVhc2UoTFBDTEFTU0ZBQ1RPUlkgaWZhY2UpCnsKCUlDT01fVEhJUyhJQ2xhc3NGYWN0b3J5SW1wbCxpZmFjZSk7CglUUkFDRSgiKCVwKS0+KGNvdW50PSVsdSlcbiIsVGhpcyxUaGlzLT5yZWYpOwoKCUludGVybG9ja2VkRGVjcmVtZW50KCZzaGVsbDMyX09iakNvdW50KTsKCWlmICghSW50ZXJsb2NrZWREZWNyZW1lbnQoJlRoaXMtPnJlZikpIAoJewoJICBUUkFDRSgiLS0gZGVzdHJveWluZyBJQ2xhc3NGYWN0b3J5KCVwKVxuIixUaGlzKTsKCSAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwwLFRoaXMpOwoJICByZXR1cm4gMDsKCX0KCXJldHVybiBUaGlzLT5yZWY7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJQ2xhc3NGYWN0b3J5X0NyZWF0ZUluc3RhbmNlCiAqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUNsYXNzRmFjdG9yeV9mbkNyZWF0ZUluc3RhbmNlKAogIExQQ0xBU1NGQUNUT1JZIGlmYWNlLCBMUFVOS05PV04gcFVua25vd24sIFJFRklJRCByaWlkLCBMUFZPSUQgKnBwT2JqZWN0KQp7CglJQ09NX1RISVMoSUNsYXNzRmFjdG9yeUltcGwsaWZhY2UpOwoJSVVua25vd24gKnBPYmogPSBOVUxMOwoJSFJFU1VMVCBocmVzOwoKCVRSQUNFKCIlcC0+KCVwLFxuXHRJSUQ6XHQlcywlcClcbiIsVGhpcyxwVW5rbm93bixkZWJ1Z3N0cl9ndWlkKHJpaWQpLHBwT2JqZWN0KTsKCgkqcHBPYmplY3QgPSBOVUxMOwoJCQoJaWYocFVua25vd24pCgl7CgkgIHJldHVybihDTEFTU19FX05PQUdHUkVHQVRJT04pOwoJfQoKCWlmIChJc0VxdWFsQ0xTSUQoVGhpcy0+cmNsc2lkLCAmQ0xTSURfU2hlbGxEZXNrdG9wKSkKCXsKCSAgcE9iaiA9IChJVW5rbm93biAqKUlTRl9EZXNrdG9wX0NvbnN0cnVjdG9yKCk7Cgl9CgllbHNlIGlmIChJc0VxdWFsQ0xTSUQoVGhpcy0+cmNsc2lkLCAmQ0xTSURfU2hlbGxMaW5rKSkKCXsKCSAgcE9iaiA9IChJVW5rbm93biAqKUlTaGVsbExpbmtfQ29uc3RydWN0b3IoRkFMU0UpOwoJfSAKCWVsc2UKCXsKCSAgRVJSKCJ1bmtub3duIElJRCByZXF1ZXN0ZWRcblx0SUlEOlx0JXNcbiIsZGVidWdzdHJfZ3VpZChyaWlkKSk7CgkgIHJldHVybihFX05PSU5URVJGQUNFKTsKCX0KCQoJaWYgKCFwT2JqKQoJewoJICByZXR1cm4oRV9PVVRPRk1FTU9SWSk7Cgl9CgkgCglocmVzID0gSVVua25vd25fUXVlcnlJbnRlcmZhY2UocE9iaixyaWlkLCBwcE9iamVjdCk7CglJVW5rbm93bl9SZWxlYXNlKHBPYmopOwoKCVRSQUNFKCItLSBPYmplY3QgY3JlYXRlZDogKCVwKS0+JXBcbiIsVGhpcywqcHBPYmplY3QpOwoKCXJldHVybiBocmVzOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSUNsYXNzRmFjdG9yeV9Mb2NrU2VydmVyCiAqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSUNsYXNzRmFjdG9yeV9mbkxvY2tTZXJ2ZXIoTFBDTEFTU0ZBQ1RPUlkgaWZhY2UsIEJPT0wgZkxvY2spCnsKCUlDT01fVEhJUyhJQ2xhc3NGYWN0b3J5SW1wbCxpZmFjZSk7CglUUkFDRSgiJXAtPigweCV4KSwgbm90IGltcGxlbWVudGVkXG4iLFRoaXMsIGZMb2NrKTsKCXJldHVybiBFX05PVElNUEw7Cn0KCnN0YXRpYyBJQ09NX1ZUQUJMRShJQ2xhc3NGYWN0b3J5KSBjbGZ2dCA9IAp7CiAgICBJQ09NX01TVlRBQkxFX0NPTVBBVF9EdW1teVJUVElWQUxVRQogICAgSUNsYXNzRmFjdG9yeV9mblF1ZXJ5SW50ZXJmYWNlLAogICAgSUNsYXNzRmFjdG9yeV9mbkFkZFJlZiwKICBJQ2xhc3NGYWN0b3J5X2ZuUmVsZWFzZSwKICBJQ2xhc3NGYWN0b3J5X2ZuQ3JlYXRlSW5zdGFuY2UsCiAgSUNsYXNzRmFjdG9yeV9mbkxvY2tTZXJ2ZXIKfTsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBEZWZhdWx0IENsYXNzRmFjdG9yeSBJbXBsZW1lbnRhdGlvbgogKgogKiBTSENyZWF0ZURlZkNsYXNzT2JqZWN0CiAqCiAqIE5PVEVTCiAqICBoZWxwZXIgZnVuY3Rpb24gZm9yIGRsbCdzIHdpdGhvdXQgYSBvd24gY2xhc3NmYWN0b3J5CiAqICBhIGdlbmVyaWMgY2xhc3NmYWN0b3J5IGlzIHJldHVybmVkCiAqICB3aGVuIHRoZSBDcmVhdGVJbnN0YW5jZSBvZiB0aGUgY2YgaXMgY2FsbGVkIHRoZSBjYWxsYmFjayBpcyBleGVjdXRlZAogKi8KdHlwZWRlZiBIUkVTVUxUIChDQUxMQkFDSyAqIExQRk5DUkVBVEVJTlNUQU5DRSkoSVVua25vd24qIHBVbmtPdXRlciwgUkVGSUlEIHJpaWQsIExQVk9JRCogcHB2T2JqZWN0KTsKCnR5cGVkZWYgc3RydWN0CnsKICAgIElDT01fVkZJRUxEKElDbGFzc0ZhY3RvcnkpOwogICAgRFdPUkQgICAgICAgICAgICAgICAgICAgICAgIHJlZjsKICAgIENMU0lECQkJKnJjbHNpZDsKICAgIExQRk5DUkVBVEVJTlNUQU5DRQkJbHBmbkNJOwogICAgY29uc3QgSUlEICoJCQlyaWlkSW5zdDsKICAgIFVMT05HICoJCQlwY1JlZkRsbDsgLyogcG9pbnRlciB0byByZWZjb3VudGVyIGluIGV4dGVybmFsIGRsbCAodWdycnIuLi4pICovCn0gSURlZkNsRkltcGw7CgpzdGF0aWMgSUNPTV9WVEFCTEUoSUNsYXNzRmFjdG9yeSkgZGNsZnZ0OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICBJRGVmQ2xGX2ZuQ29uc3RydWN0b3IKICovCgpJQ2xhc3NGYWN0b3J5ICogSURlZkNsRl9mbkNvbnN0cnVjdG9yKExQRk5DUkVBVEVJTlNUQU5DRSBscGZuQ0ksIFBMT05HIHBjUmVmRGxsLCBSRUZJSUQgcmlpZEluc3QpCnsKCUlEZWZDbEZJbXBsKiBscGNsZjsKCglscGNsZiA9IChJRGVmQ2xGSW1wbCopSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksMCxzaXplb2YoSURlZkNsRkltcGwpKTsKCWxwY2xmLT5yZWYgPSAxOwoJSUNPTV9WVEJMKGxwY2xmKSA9ICZkY2xmdnQ7CglscGNsZi0+bHBmbkNJID0gbHBmbkNJOwoJbHBjbGYtPnBjUmVmRGxsID0gcGNSZWZEbGw7CgoJaWYgKHBjUmVmRGxsKSBJbnRlcmxvY2tlZEluY3JlbWVudChwY1JlZkRsbCk7CglscGNsZi0+cmlpZEluc3QgPSByaWlkSW5zdDsKCglUUkFDRSgiKCVwKVxuXHRJSUQ6XHQlc1xuIixscGNsZiwgZGVidWdzdHJfZ3VpZChyaWlkSW5zdCkpOwoJSW50ZXJsb2NrZWRJbmNyZW1lbnQoJnNoZWxsMzJfT2JqQ291bnQpOwoJcmV0dXJuIChMUENMQVNTRkFDVE9SWSlscGNsZjsKfQovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogIElEZWZDbEZfZm5RdWVyeUludGVyZmFjZQogKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJIElEZWZDbEZfZm5RdWVyeUludGVyZmFjZSgKICBMUENMQVNTRkFDVE9SWSBpZmFjZSwgUkVGSUlEIHJpaWQsIExQVk9JRCAqcHB2T2JqKQp7CglJQ09NX1RISVMoSURlZkNsRkltcGwsaWZhY2UpOwoKCVRSQUNFKCIoJXApLT4oXG5cdElJRDpcdCVzKVxuIixUaGlzLGRlYnVnc3RyX2d1aWQocmlpZCkpOwoKCSpwcHZPYmogPSBOVUxMOwoKCWlmKElzRXF1YWxJSUQocmlpZCwgJklJRF9JVW5rbm93bikpICAgICAgICAgIC8qSVVua25vd24qLwoJeyAqcHB2T2JqID0gVGhpczsgCgl9CgllbHNlIGlmKElzRXF1YWxJSUQocmlpZCwgJklJRF9JQ2xhc3NGYWN0b3J5KSkgIC8qSUNsYXNzRmFjdG9yeSovCgl7ICpwcHZPYmogPSAoSUNsYXNzRmFjdG9yeSopVGhpczsKCX0gICAKCglpZigqcHB2T2JqKQoJeyBJVW5rbm93bl9BZGRSZWYoKExQVU5LTk9XTikqcHB2T2JqKTsgIAkKCSAgVFJBQ0UoIi0tIEludGVyZmFjZTogKCVwKS0+KCVwKVxuIixwcHZPYmosKnBwdk9iaik7CgkgIHJldHVybiBTX09LOwoJfQoJVFJBQ0UoIi0tIEludGVyZmFjZTogJXMgRV9OT0lOVEVSRkFDRVxuIiwgZGVidWdzdHJfZ3VpZChyaWlkKSk7CglyZXR1cm4gRV9OT0lOVEVSRkFDRTsKfSAgCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURlZkNsRl9mbkFkZFJlZgogKi8Kc3RhdGljIFVMT05HIFdJTkFQSSBJRGVmQ2xGX2ZuQWRkUmVmKExQQ0xBU1NGQUNUT1JZIGlmYWNlKQp7CglJQ09NX1RISVMoSURlZkNsRkltcGwsaWZhY2UpOwoJVFJBQ0UoIiglcCktPihjb3VudD0lbHUpXG4iLFRoaXMsVGhpcy0+cmVmKTsKCglJbnRlcmxvY2tlZEluY3JlbWVudCgmc2hlbGwzMl9PYmpDb3VudCk7CglyZXR1cm4gSW50ZXJsb2NrZWRJbmNyZW1lbnQoJlRoaXMtPnJlZik7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGVmQ2xGX2ZuUmVsZWFzZQogKi8Kc3RhdGljIFVMT05HIFdJTkFQSSBJRGVmQ2xGX2ZuUmVsZWFzZShMUENMQVNTRkFDVE9SWSBpZmFjZSkKewoJSUNPTV9USElTKElEZWZDbEZJbXBsLGlmYWNlKTsKCVRSQUNFKCIoJXApLT4oY291bnQ9JWx1KVxuIixUaGlzLFRoaXMtPnJlZik7CgoJSW50ZXJsb2NrZWREZWNyZW1lbnQoJnNoZWxsMzJfT2JqQ291bnQpOwoKCWlmICghSW50ZXJsb2NrZWREZWNyZW1lbnQoJlRoaXMtPnJlZikpIAoJeyAKCSAgaWYgKFRoaXMtPnBjUmVmRGxsKSBJbnRlcmxvY2tlZERlY3JlbWVudChUaGlzLT5wY1JlZkRsbCk7CgoJICBUUkFDRSgiLS0gZGVzdHJveWluZyBJQ2xhc3NGYWN0b3J5KCVwKVxuIixUaGlzKTsKCSAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwwLFRoaXMpOwoJICByZXR1cm4gMDsKCX0KCXJldHVybiBUaGlzLT5yZWY7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGVmQ2xGX2ZuQ3JlYXRlSW5zdGFuY2UKICovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJRGVmQ2xGX2ZuQ3JlYXRlSW5zdGFuY2UoCiAgTFBDTEFTU0ZBQ1RPUlkgaWZhY2UsIExQVU5LTk9XTiBwVW5rT3V0ZXIsIFJFRklJRCByaWlkLCBMUFZPSUQgKnBwdk9iamVjdCkKewoJSUNPTV9USElTKElEZWZDbEZJbXBsLGlmYWNlKTsKCglUUkFDRSgiJXAtPiglcCxcblx0SUlEOlx0JXMsJXApXG4iLFRoaXMscFVua091dGVyLGRlYnVnc3RyX2d1aWQocmlpZCkscHB2T2JqZWN0KTsKCgkqcHB2T2JqZWN0ID0gTlVMTDsKCQkKCWlmKHBVbmtPdXRlcikKCSAgcmV0dXJuKENMQVNTX0VfTk9BR0dSRUdBVElPTik7CgoJaWYgKCBUaGlzLT5yaWlkSW5zdD09TlVMTCB8fAoJICAgICBJc0VxdWFsQ0xTSUQocmlpZCwgVGhpcy0+cmlpZEluc3QpIHx8CgkgICAgIElzRXF1YWxDTFNJRChyaWlkLCAmSUlEX0lVbmtub3duKSApCgl7CgkgIHJldHVybiBUaGlzLT5scGZuQ0kocFVua091dGVyLCByaWlkLCBwcHZPYmplY3QpOwoJfQoKCUVSUigidW5rbm93biBJSUQgcmVxdWVzdGVkXG5cdElJRDpcdCVzXG4iLGRlYnVnc3RyX2d1aWQocmlpZCkpOwoJcmV0dXJuIEVfTk9JTlRFUkZBQ0U7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJRGVmQ2xGX2ZuTG9ja1NlcnZlcgogKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJIElEZWZDbEZfZm5Mb2NrU2VydmVyKExQQ0xBU1NGQUNUT1JZIGlmYWNlLCBCT09MIGZMb2NrKQp7CglJQ09NX1RISVMoSURlZkNsRkltcGwsaWZhY2UpOwoJVFJBQ0UoIiVwLT4oMHgleCksIG5vdCBpbXBsZW1lbnRlZFxuIixUaGlzLCBmTG9jayk7CglyZXR1cm4gRV9OT1RJTVBMOwp9CgpzdGF0aWMgSUNPTV9WVEFCTEUoSUNsYXNzRmFjdG9yeSkgZGNsZnZ0ID0gCnsKICAgIElDT01fTVNWVEFCTEVfQ09NUEFUX0R1bW15UlRUSVZBTFVFCiAgICBJRGVmQ2xGX2ZuUXVlcnlJbnRlcmZhY2UsCiAgICBJRGVmQ2xGX2ZuQWRkUmVmLAogIElEZWZDbEZfZm5SZWxlYXNlLAogIElEZWZDbEZfZm5DcmVhdGVJbnN0YW5jZSwKICBJRGVmQ2xGX2ZuTG9ja1NlcnZlcgp9OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBTSENyZWF0ZURlZkNsYXNzT2JqZWN0CQkJW1NIRUxMMzIuNzBdCiAqLwpIUkVTVUxUIFdJTkFQSSBTSENyZWF0ZURlZkNsYXNzT2JqZWN0KAoJUkVGSUlECXJpaWQsCQkJCQoJTFBWT0lEKglwcHYsCQoJTFBGTkNSRUFURUlOU1RBTkNFIGxwZm5DSSwJLyogY3JlYXRlIGluc3RhbmNlIGNhbGxiYWNrIGVudHJ5ICovCglQTE9ORwlwY1JlZkRsbCwJCS8qIHJlZiBjb3VudCBvZiB0aGUgZGxsICovCglSRUZJSUQJcmlpZEluc3QpCQkvKiBvcHRpb25hbCBpbnRlcmZhY2UgdG8gdGhlIGluc3RhbmNlICovCnsKCVRSQUNFKCJcblx0SUlEOlx0JXMgJXAgJXAgJXAgXG5cdElJREluczpcdCVzXG4iLAogICAgICAgICAgICAgIGRlYnVnc3RyX2d1aWQocmlpZCksIHBwdiwgbHBmbkNJLCBwY1JlZkRsbCwgZGVidWdzdHJfZ3VpZChyaWlkSW5zdCkpOwoKCWlmICggSXNFcXVhbENMU0lEKHJpaWQsICZJSURfSUNsYXNzRmFjdG9yeSkgKQoJewoJICBJQ2xhc3NGYWN0b3J5ICogcGNmID0gSURlZkNsRl9mbkNvbnN0cnVjdG9yKGxwZm5DSSwgcGNSZWZEbGwsIHJpaWRJbnN0KTsKCSAgaWYgKHBjZikKCSAgewoJICAgICpwcHYgPSBwY2Y7CgkgICAgcmV0dXJuIE5PRVJST1I7CgkgIH0KCSAgcmV0dXJuIEVfT1VUT0ZNRU1PUlk7Cgl9CglyZXR1cm4gRV9OT0lOVEVSRkFDRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogIERyYWdBY2NlcHRGaWxlcwkJW1NIRUxMMzIuNTRdCiAqLwp2b2lkIFdJTkFQSSBEcmFnQWNjZXB0RmlsZXMoSFdORCBoV25kLCBCT09MIGIpCnsKCUxPTkcgZXhzdHlsZTsKICAKCWlmKCAhSXNXaW5kb3coaFduZCkgKSByZXR1cm47CglleHN0eWxlID0gR2V0V2luZG93TG9uZ0EoaFduZCxHV0xfRVhTVFlMRSk7CglpZiAoYikKCSAgZXhzdHlsZSB8PSBXU19FWF9BQ0NFUFRGSUxFUzsKCWVsc2UKCSAgZXhzdHlsZSAmPSB+V1NfRVhfQUNDRVBURklMRVM7CglTZXRXaW5kb3dMb25nQShoV25kLEdXTF9FWFNUWUxFLGV4c3R5bGUpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBEcmFnRmluaXNoCQlbU0hFTEwzMi44MF0KICovCnZvaWQgV0lOQVBJIERyYWdGaW5pc2goSERST1AgaCkKewoJVFJBQ0UoIlxuIik7CglHbG9iYWxGcmVlKChIR0xPQkFMKWgpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBEcmFnUXVlcnlQb2ludAkJW1NIRUxMMzIuMTM1XQogKi8KQk9PTCBXSU5BUEkgRHJhZ1F1ZXJ5UG9pbnQoSERST1AgaERyb3AsIFBPSU5UICpwKQp7CglMUERST1BGSUxFU1RSVUNUIGxwRHJvcEZpbGVTdHJ1Y3Q7ICAKCUJPT0wgYlJldDsKCglUUkFDRSgiXG4iKTsKCglscERyb3BGaWxlU3RydWN0ID0gKExQRFJPUEZJTEVTVFJVQ1QpIEdsb2JhbExvY2soaERyb3ApOwogIAoJbWVtY3B5KHAsJmxwRHJvcEZpbGVTdHJ1Y3QtPnB0TW91c2VQb3Msc2l6ZW9mKFBPSU5UKSk7CgliUmV0ID0gbHBEcm9wRmlsZVN0cnVjdC0+ZkluTm9uQ2xpZW50QXJlYTsKICAKCUdsb2JhbFVubG9jayhoRHJvcCk7CglyZXR1cm4gYlJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogIERyYWdRdWVyeUZpbGVBCQlbU0hFTEwzMi44MV0gW3NoZWxsMzIuODJdCiAqLwpVSU5UIFdJTkFQSSBEcmFnUXVlcnlGaWxlQSgKCUhEUk9QIGhEcm9wLAoJVUlOVCBsRmlsZSwKCUxQU1RSIGxwc3pGaWxlLAoJVUlOVCBsTGVuZ3RoKQp7CglMUFNUUiBscERyb3A7CglVSU5UIGkgPSAwOwoJTFBEUk9QRklMRVNUUlVDVCBscERyb3BGaWxlU3RydWN0ID0gKExQRFJPUEZJTEVTVFJVQ1QpIEdsb2JhbExvY2soaERyb3ApOyAKICAgIAoJVFJBQ0UoIiglMDh4LCAleCwgJXAsICV1KVxuIiwJaERyb3AsbEZpbGUsbHBzekZpbGUsbExlbmd0aCk7CiAgICAKCWlmKCFscERyb3BGaWxlU3RydWN0KSBnb3RvIGVuZDsKCglscERyb3AgPSAoTFBTVFIpIGxwRHJvcEZpbGVTdHJ1Y3QgKyBscERyb3BGaWxlU3RydWN0LT5sU2l6ZTsKCgl3aGlsZSAoaSsrIDwgbEZpbGUpCgl7CgkgIHdoaWxlICgqbHBEcm9wKyspOyAvKiBza2lwIGZpbGVuYW1lICovCgkgIGlmICghKmxwRHJvcCkgCgkgIHsKCSAgICBpID0gKGxGaWxlID09IDB4RkZGRkZGRkYpID8gaSA6IDA7IAoJICAgIGdvdG8gZW5kOwoJICB9Cgl9CiAgICAKCWkgPSBsc3RybGVuQShscERyb3ApOwoJaSsrOwoJaWYgKCFscHN6RmlsZSApIGdvdG8gZW5kOyAgIC8qIG5lZWRlZCBidWZmZXIgc2l6ZSAqLwoJaSA9IChsTGVuZ3RoID4gaSkgPyBpIDogbExlbmd0aDsKCWxzdHJjcHluQSAobHBzekZpbGUsICBscERyb3AsICBpKTsKZW5kOgoJR2xvYmFsVW5sb2NrKGhEcm9wKTsKCXJldHVybiBpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgRHJhZ1F1ZXJ5RmlsZVcJCVtzaGVsbDMyLjEzM10KICovClVJTlQgV0lOQVBJIERyYWdRdWVyeUZpbGVXKAoJSERST1AgaERyb3AsCglVSU5UIGxGaWxlLAoJTFBXU1RSIGxwc3p3RmlsZSwKCVVJTlQgbExlbmd0aCkKewoJTFBXU1RSIGxwd0Ryb3A7CglVSU5UIGkgPSAwOwoJTFBEUk9QRklMRVNUUlVDVCBscERyb3BGaWxlU3RydWN0ID0gKExQRFJPUEZJTEVTVFJVQ1QpIEdsb2JhbExvY2soaERyb3ApOyAKICAgIAoJVFJBQ0UoIiglMDh4LCAleCwgJXAsICV1KVxuIiwgaERyb3AsbEZpbGUsbHBzendGaWxlLGxMZW5ndGgpOwogICAgCglpZighbHBEcm9wRmlsZVN0cnVjdCkgZ290byBlbmQ7CgoJbHB3RHJvcCA9IChMUFdTVFIpIGxwRHJvcEZpbGVTdHJ1Y3QgKyBscERyb3BGaWxlU3RydWN0LT5sU2l6ZTsKCglpID0gMDsKCXdoaWxlIChpKysgPCBsRmlsZSkKCXsKCSAgd2hpbGUgKCpscHdEcm9wKyspOyAvKiBza2lwIGZpbGVuYW1lICovCgkgIGlmICghKmxwd0Ryb3ApIAoJICB7CgkgICAgaSA9IChsRmlsZSA9PSAweEZGRkZGRkZGKSA/IGkgOiAwOyAKCSAgICBnb3RvIGVuZDsKCSAgfQoJfQogICAgCglpID0gbHN0cmxlblcobHB3RHJvcCk7CglpKys7CglpZiAoICFscHN6d0ZpbGUpIGdvdG8gZW5kOyAgIC8qIG5lZWRlZCBidWZmZXIgc2l6ZSAqLwoKCWkgPSAobExlbmd0aCA+IGkpID8gaSA6IGxMZW5ndGg7Cglsc3RyY3B5blcgKGxwc3p3RmlsZSwgbHB3RHJvcCwgaSk7CmVuZDoKCUdsb2JhbFVubG9jayhoRHJvcCk7CglyZXR1cm4gaTsKfQo=