LyoKICoJaGFuZGxpbmcgb2YgU0hFTEwzMi5ETEwgT0xFLU9iamVjdHMKICoKICoJQ29weXJpZ2h0IDE5OTcJTWFyY3VzIE1laXNzbmVyCiAqCUNvcHlyaWdodCAxOTk4CUp1ZXJnZW4gU2NobWllZCAgPGp1ZXJnZW4uc2NobWllZEBtZXRyb25ldC5kZT4KICoKICovCgojaW5jbHVkZSA8c3RkbGliLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KCiNpbmNsdWRlICJ3aW5lL29ial9iYXNlLmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGxpbmsuaCIKI2luY2x1ZGUgIndpbmUvb2JqX3NoZWxsZm9sZGVyLmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGJyb3dzZXIuaCIKI2luY2x1ZGUgIndpbmUvb2JqX2NvbnRleHRtZW51LmgiCiNpbmNsdWRlICJ3aW5lL29ial9zaGVsbGV4dGluaXQuaCIKI2luY2x1ZGUgIndpbmUvb2JqX2V4dHJhY3RpY29uLmgiCgojaW5jbHVkZSAic2hsZ3VpZC5oIgojaW5jbHVkZSAid2ludmVyc2lvbi5oIgojaW5jbHVkZSAid2lucmVnLmgiCiNpbmNsdWRlICJ3aW5lcnJvci5oIgojaW5jbHVkZSAiZGVidWd0b29scy5oIgoKI2luY2x1ZGUgInNoZWxsMzJfbWFpbi5oIgoKREVGQVVMVF9ERUJVR19DSEFOTkVMKHNoZWxsKQoKRFdPUkQgV0lOQVBJIFNIQ0xTSURGcm9tU3RyaW5nQSAoTFBTVFIgY2xzaWQsIENMU0lEICppZCk7CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBTSENvQ3JlYXRlSW5zdGFuY2UgW1NIRUxMMzIuMTAyXQogKiAKICogTk9URVMKICogICAgIGV4cG9ydGVkIGJ5IG9yZGluYWwKICovCkxSRVNVTFQgV0lOQVBJIFNIQ29DcmVhdGVJbnN0YW5jZSgKCUxQU1RSIGFjbHNpZCwKCVJFRkNMU0lEIGNsc2lkLAoJSVVua25vd24gKiB1bmtub3dub3V0ZXIsCglSRUZJSUQgcmVmaWlkLAoJTFBWT0lEICpwcHYpCnsKCWNoYXIJeGNsc2lkWzQ4XSwgeGlpZFs0OF0sIHh1b3V0WzQ4XTsKCURXT1JECWhyZXM7CglJSUQJaWlkOwoJQ0xTSUQgKiBteWNsc2lkID0gKENMU0lEKiljbHNpZDsKCQoJV0lORV9TdHJpbmdGcm9tQ0xTSUQocmVmaWlkLHhpaWQpOwoKCWlmICghY2xzaWQpCgl7CgkgIGlmICghYWNsc2lkKSByZXR1cm4gUkVHREJfRV9DTEFTU05PVFJFRzsKCSAgU0hDTFNJREZyb21TdHJpbmdBKGFjbHNpZCwgJmlpZCk7CgkgIG15Y2xzaWQgPSAmaWlkOwoJfQoKCVdJTkVfU3RyaW5nRnJvbUNMU0lEKG15Y2xzaWQseGNsc2lkKTsKCVdJTkVfU3RyaW5nRnJvbUNMU0lEKHJlZmlpZCx4aWlkKTsKCWlmICh1bmtub3dub3V0ZXIpCgkJV0lORV9TdHJpbmdGcm9tQ0xTSUQodW5rbm93bm91dGVyLHh1b3V0KTsKCSAgCglUUkFDRSgiKCVwLFxuXHRDTFNJRDpcdCVzXG5cdFVPVVQ6XHQlc1xuXHRJSUQ6XHQlcywlcClcbiIsCgkJYWNsc2lkLHhjbHNpZCx1bmtub3dub3V0ZXI/eHVvdXQ6Im5pbCIseGlpZCxwcHYpOwoKCWhyZXMgPSBDb0NyZWF0ZUluc3RhbmNlKG15Y2xzaWQsIE5VTEwsIENMU0NUWF9JTlBST0NfU0VSVkVSLCByZWZpaWQsIHBwdik7CgoJaWYoaHJlcyE9U19PSykKCXsKCSAgRVJSKCJmYWlsZWQgKDB4JTA4bHgpIHRvIGNyZWF0ZSBcblx0Q0xTSUQ6XHQlc1xuXHRJSUQ6XHQlc1xuIiwgaHJlcywgeGNsc2lkLCB4aWlkKTsKCSAgRVJSKCJ5b3UgbWlnaHQgbmVlZCB0byBpbXBvcnQgdGhlIHdpbmVkZWZhdWx0LnJlZ1xuIik7Cgl9CgoJcmV0dXJuIGhyZXM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIFNIRUxMMzJfRGxsR2V0Q2xhc3NPYmplY3QgICBbU0hFTEwzMi4xMjhdCiAqLwpIUkVTVUxUIFdJTkFQSSBTSEVMTDMyX0RsbEdldENsYXNzT2JqZWN0KFJFRkNMU0lEIHJjbHNpZCwgUkVGSUlEIGlpZCxMUFZPSUQgKnBwdikKewlIUkVTVUxUCWhyZXMgPSBFX09VVE9GTUVNT1JZOwoJTFBDTEFTU0ZBQ1RPUlkgbHBjbGY7CgoJY2hhciB4Y2xzaWRbNTBdLHhpaWRbNTBdOwoJV0lORV9TdHJpbmdGcm9tQ0xTSUQoKExQQ0xTSUQpcmNsc2lkLHhjbHNpZCk7CglXSU5FX1N0cmluZ0Zyb21DTFNJRCgoTFBDTFNJRClpaWQseGlpZCk7CglUUkFDRSgiXG5cdENMU0lEOlx0JXMsXG5cdElJRDpcdCVzXG4iLHhjbHNpZCx4aWlkKTsKCQoJKnBwdiA9IE5VTEw7CgoJaWYoSXNFcXVhbENMU0lEKHJjbHNpZCwgJkNMU0lEX1BhcGVyQmluKSkKCXsKCSAgRVJSKCJwYXBlciBiaW4gbm90IGltcGxlbWVudGVkXG4iKTsKCSAgcmV0dXJuIENMQVNTX0VfQ0xBU1NOT1RBVkFJTEFCTEU7Cgl9CglpZihJc0VxdWFsQ0xTSUQocmNsc2lkLCAmQ0xTSURfU2hlbGxEZXNrdG9wKXx8IAoJICAgSXNFcXVhbENMU0lEKHJjbHNpZCwgJkNMU0lEX1NoZWxsTGluaykpCgl7CgkgIGxwY2xmID0gSUNsYXNzRmFjdG9yeV9Db25zdHJ1Y3RvciggcmNsc2lkICk7CgoJICBpZihscGNsZikgCgkgIHsKCSAgICBocmVzID0gSUNsYXNzRmFjdG9yeV9RdWVyeUludGVyZmFjZShscGNsZixpaWQsIHBwdik7CgkgICAgSUNsYXNzRmFjdG9yeV9SZWxlYXNlKGxwY2xmKTsKCSAgfQoJfQoJZWxzZQoJewoJICBXQVJOKCItLSBDTFNJRCBub3QgZm91bmRcbiIpOwoJICBocmVzID0gQ0xBU1NfRV9DTEFTU05PVEFWQUlMQUJMRTsKCX0KCVRSQUNFKCItLSBwb2ludGVyIHRvIGNsYXNzIGZhY3Rvcnk6ICVwXG4iLCpwcHYpOwoJcmV0dXJuIGhyZXM7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIFNIQ0xTSURGcm9tU3RyaW5nCQkJCVtTSEVMTDMyLjE0N10KICoKICogTk9URVMKICogICAgIGV4cG9ydGVkIGJ5IG9yZGluYWwKICovCkRXT1JEIFdJTkFQSSBTSENMU0lERnJvbVN0cmluZ0EgKExQU1RSIGNsc2lkLCBDTFNJRCAqaWQpCnsKCVRSQUNFKCIoJXAoJXMpICVwKVxuIiwgY2xzaWQsIGNsc2lkLCBpZCk7CglyZXR1cm4gQ0xTSURGcm9tU3RyaW5nMTYoY2xzaWQsIGlkKTsgCn0KRFdPUkQgV0lOQVBJIFNIQ0xTSURGcm9tU3RyaW5nVyAoTFBXU1RSIGNsc2lkLCBDTFNJRCAqaWQpCnsKCVRSQUNFKCIoJXAoJXMpICVwKVxuIiwgY2xzaWQsIGRlYnVnc3RyX3coY2xzaWQpLCBpZCk7CglyZXR1cm4gQ0xTSURGcm9tU3RyaW5nKGNsc2lkLCBpZCk7IAp9CkRXT1JEIFdJTkFQSSBTSENMU0lERnJvbVN0cmluZ0FXIChMUFZPSUQgY2xzaWQsIENMU0lEICppZCkKewoJaWYgKFZFUlNJT05fT3NJc1VuaWNvZGUoKSkKCSAgcmV0dXJuIFNIQ0xTSURGcm9tU3RyaW5nVyAoY2xzaWQsIGlkKTsKCXJldHVybiBTSENMU0lERnJvbVN0cmluZ0EgKGNsc2lkLCBpZCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQkJIFNIR2V0TWFsbG9jCQkJW1NIRUxMMzIuMjIwXQogKiByZXR1cm5zIHRoZSBpbnRlcmZhY2UgdG8gc2hlbGwgbWFsbG9jLgogKgogKiBbU0RLIGhlYWRlciB3aW45NS9zaGxvYmouaDoKICogZXF1aXZhbGVudCB0bzogICNkZWZpbmUgU0hHZXRNYWxsb2MocHBtZW0pICAgQ29HZXRNYWxsb2MoTUVNQ1RYX1RBU0ssIHBwbWVtKQogKiBdCiAqIFdoYXQgd2UgYXJlIGN1cnJlbnRseSBkb2luZyBpcyBub3QgdmVyeSB3cm9uZywgc2luY2Ugd2UgYWx3YXlzIHVzZSB0aGUgc2FtZQogKiBoZWFwIChQcm9jZXNzSGVhcCkuCiAqLwpEV09SRCBXSU5BUEkgU0hHZXRNYWxsb2MoTFBNQUxMT0MgKmxwbWFsKSAKewoJVFJBQ0UoIiglcClcbiIsIGxwbWFsKTsKCXJldHVybiBDb0dldE1hbGxvYygwLGxwbWFsKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogU0hHZXREZXNrdG9wRm9sZGVyCQkJW1NIRUxMMzIuMjE2XQogKi8KTFBTSEVMTEZPTERFUiBwZGVza3RvcGZvbGRlcj1OVUxMOwoKRFdPUkQgV0lOQVBJIFNIR2V0RGVza3RvcEZvbGRlcihJU2hlbGxGb2xkZXIgKipwc2YpCnsKCUhSRVNVTFQJaHJlcyA9IFNfT0s7CglMUENMQVNTRkFDVE9SWSBscGNsZjsKCVRSQUNFXyhzaGVsbCkoIiVwLT4oJXApXG4iLHBzZiwqcHNmKTsKCgkqcHNmPU5VTEw7CgoJaWYgKCFwZGVza3RvcGZvbGRlcikgCgl7CgkgIGxwY2xmID0gSUNsYXNzRmFjdG9yeV9Db25zdHJ1Y3RvcigmQ0xTSURfU2hlbGxEZXNrdG9wKTsKCSAgaWYobHBjbGYpIAoJICB7CgkgICAgaHJlcyA9IElDbGFzc0ZhY3RvcnlfQ3JlYXRlSW5zdGFuY2UobHBjbGYsTlVMTCwoUkVGSUlEKSZJSURfSVNoZWxsRm9sZGVyLCAodm9pZCopJnBkZXNrdG9wZm9sZGVyKTsKCSAgICBJQ2xhc3NGYWN0b3J5X1JlbGVhc2UobHBjbGYpOwoJICB9ICAKCX0KCQoJaWYgKHBkZXNrdG9wZm9sZGVyKSAKCXsKCSAgLyogZXZlbiBpZiB3ZSBjcmVhdGUgdGhlIGZvbGRlciwgYWRkIGEgcmVmIHNvIHRoZSBhcHBsaWNhdGlvbiBjYW60dCBkZXN0cm95IHRoZSBmb2xkZXIqLwoJICBJU2hlbGxGb2xkZXJfQWRkUmVmKHBkZXNrdG9wZm9sZGVyKTsKCSAgKnBzZiA9IHBkZXNrdG9wZm9sZGVyOwoJfQoKCVRSQUNFXyhzaGVsbCkoIi0tICVwLT4oJXApXG4iLHBzZiwgKnBzZik7CglyZXR1cm4gaHJlczsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiogIElDbGFzc0ZhY3RvcnkgSW1wbGVtZW50YXRpb24KKi8KCnR5cGVkZWYgc3RydWN0CnsKICAgIC8qIElVbmtub3duIGZpZWxkcyAqLwogICAgSUNPTV9WRklFTEQoSUNsYXNzRmFjdG9yeSk7CiAgICBEV09SRCAgICAgICAgICAgICAgICAgICAgICAgcmVmOwogICAgQ0xTSUQJCQkqcmNsc2lkOwp9IElDbGFzc0ZhY3RvcnlJbXBsOwoKc3RhdGljIElDT01fVlRBQkxFKElDbGFzc0ZhY3RvcnkpIGNsZnZ0OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICBJQ2xhc3NGYWN0b3J5X0NvbnN0cnVjdG9yCiAqLwoKTFBDTEFTU0ZBQ1RPUlkgSUNsYXNzRmFjdG9yeV9Db25zdHJ1Y3RvcihSRUZDTFNJRCByY2xzaWQpCnsKCUlDbGFzc0ZhY3RvcnlJbXBsKiBscGNsZjsKCglscGNsZj0gKElDbGFzc0ZhY3RvcnlJbXBsKilIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwwLHNpemVvZihJQ2xhc3NGYWN0b3J5SW1wbCkpOwoJbHBjbGYtPnJlZiA9IDE7CglJQ09NX1ZUQkwobHBjbGYpID0gJmNsZnZ0OwoJbHBjbGYtPnJjbHNpZCA9IChDTFNJRCopcmNsc2lkOwoKCVRSQUNFKCIoJXApLT4oKVxuIixscGNsZik7CglzaGVsbDMyX09iakNvdW50Kys7CglyZXR1cm4gKExQQ0xBU1NGQUNUT1JZKWxwY2xmOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgSUNsYXNzRmFjdG9yeV9RdWVyeUludGVyZmFjZQogKi8Kc3RhdGljIEhSRVNVTFQgV0lOQVBJIElDbGFzc0ZhY3RvcnlfZm5RdWVyeUludGVyZmFjZSgKICBMUENMQVNTRkFDVE9SWSBpZmFjZSwgUkVGSUlEIHJpaWQsIExQVk9JRCAqcHB2T2JqKQp7CglJQ09NX1RISVMoSUNsYXNzRmFjdG9yeUltcGwsaWZhY2UpOwoJY2hhcgl4cmlpZFs1MF07CglXSU5FX1N0cmluZ0Zyb21DTFNJRCgoTFBDTFNJRClyaWlkLHhyaWlkKTsKCVRSQUNFKCIoJXApLT4oXG5cdElJRDpcdCVzKVxuIixUaGlzLHhyaWlkKTsKCgkqcHB2T2JqID0gTlVMTDsKCglpZihJc0VxdWFsSUlEKHJpaWQsICZJSURfSVVua25vd24pKSAgICAgICAgICAvKklVbmtub3duKi8KCXsgKnBwdk9iaiA9IFRoaXM7IAoJfQoJZWxzZSBpZihJc0VxdWFsSUlEKHJpaWQsICZJSURfSUNsYXNzRmFjdG9yeSkpICAvKklDbGFzc0ZhY3RvcnkqLwoJeyAqcHB2T2JqID0gKElDbGFzc0ZhY3RvcnkqKVRoaXM7Cgl9ICAgCgoJaWYoKnBwdk9iaikKCXsgSVVua25vd25fQWRkUmVmKChMUFVOS05PV04pKnBwdk9iaik7ICAJCgkgIFRSQUNFKCItLSBJbnRlcmZhY2U6ICglcCktPiglcClcbiIscHB2T2JqLCpwcHZPYmopOwoJICByZXR1cm4gU19PSzsKCX0KCVRSQUNFKCItLSBJbnRlcmZhY2U6ICVzIEVfTk9JTlRFUkZBQ0VcbiIsIHhyaWlkKTsKCXJldHVybiBFX05PSU5URVJGQUNFOwp9ICAKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJQ2xhc3NGYWN0b3J5X0FkZFJlZgogKi8Kc3RhdGljIFVMT05HIFdJTkFQSSBJQ2xhc3NGYWN0b3J5X2ZuQWRkUmVmKExQQ0xBU1NGQUNUT1JZIGlmYWNlKQp7CglJQ09NX1RISVMoSUNsYXNzRmFjdG9yeUltcGwsaWZhY2UpOwoJVFJBQ0UoIiglcCktPihjb3VudD0lbHUpXG4iLFRoaXMsVGhpcy0+cmVmKTsKCglzaGVsbDMyX09iakNvdW50Kys7CglyZXR1cm4gKysoVGhpcy0+cmVmKTsKfQovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElDbGFzc0ZhY3RvcnlfUmVsZWFzZQogKi8Kc3RhdGljIFVMT05HIFdJTkFQSSBJQ2xhc3NGYWN0b3J5X2ZuUmVsZWFzZShMUENMQVNTRkFDVE9SWSBpZmFjZSkKewoJSUNPTV9USElTKElDbGFzc0ZhY3RvcnlJbXBsLGlmYWNlKTsKCVRSQUNFKCIoJXApLT4oY291bnQ9JWx1KVxuIixUaGlzLFRoaXMtPnJlZik7CgoJc2hlbGwzMl9PYmpDb3VudC0tOwoJaWYgKCEtLShUaGlzLT5yZWYpKSAKCXsgVFJBQ0UoIi0tIGRlc3Ryb3lpbmcgSUNsYXNzRmFjdG9yeSglcClcbiIsVGhpcyk7CgkJSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwwLFRoaXMpOwoJCXJldHVybiAwOwoJfQoJcmV0dXJuIFRoaXMtPnJlZjsKfQovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElDbGFzc0ZhY3RvcnlfQ3JlYXRlSW5zdGFuY2UKICovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJQ2xhc3NGYWN0b3J5X2ZuQ3JlYXRlSW5zdGFuY2UoCiAgTFBDTEFTU0ZBQ1RPUlkgaWZhY2UsIExQVU5LTk9XTiBwVW5rbm93biwgUkVGSUlEIHJpaWQsIExQVk9JRCAqcHBPYmplY3QpCnsKCUlDT01fVEhJUyhJQ2xhc3NGYWN0b3J5SW1wbCxpZmFjZSk7CglJVW5rbm93biAqcE9iaiA9IE5VTEw7CglIUkVTVUxUIGhyZXM7CgljaGFyCXhyaWlkWzUwXTsKCglXSU5FX1N0cmluZ0Zyb21DTFNJRCgoTFBDTFNJRClyaWlkLHhyaWlkKTsKCVRSQUNFKCIlcC0+KCVwLFxuXHRJSUQ6XHQlcywlcClcbiIsVGhpcyxwVW5rbm93bix4cmlpZCxwcE9iamVjdCk7CgoJKnBwT2JqZWN0ID0gTlVMTDsKCQkKCWlmKHBVbmtub3duKQoJewoJICByZXR1cm4oQ0xBU1NfRV9OT0FHR1JFR0FUSU9OKTsKCX0KCglpZiAoSXNFcXVhbENMU0lEKFRoaXMtPnJjbHNpZCwgJkNMU0lEX1NoZWxsRGVza3RvcCkpCgl7CgkgIHBPYmogPSAoSVVua25vd24gKilJU0ZfRGVza3RvcF9Db25zdHJ1Y3RvcigpOwoJfQoJZWxzZSBpZiAoSXNFcXVhbENMU0lEKFRoaXMtPnJjbHNpZCwgJkNMU0lEX1NoZWxsTGluaykpCgl7CgkgIHBPYmogPSAoSVVua25vd24gKilJU2hlbGxMaW5rX0NvbnN0cnVjdG9yKEZBTFNFKTsKCX0gCgllbHNlCgl7CgkgIEVSUigidW5rbm93biBJSUQgcmVxdWVzdGVkXG5cdElJRDpcdCVzXG4iLHhyaWlkKTsKCSAgcmV0dXJuKEVfTk9JTlRFUkZBQ0UpOwoJfQoJCglpZiAoIXBPYmopCgl7CgkgIHJldHVybihFX09VVE9GTUVNT1JZKTsKCX0KCSAKCWhyZXMgPSBJVW5rbm93bl9RdWVyeUludGVyZmFjZShwT2JqLHJpaWQsIHBwT2JqZWN0KTsKCUlVbmtub3duX1JlbGVhc2UocE9iaik7CgoJVFJBQ0UoIi0tIE9iamVjdCBjcmVhdGVkOiAoJXApLT4lcFxuIixUaGlzLCpwcE9iamVjdCk7CgoJcmV0dXJuIGhyZXM7Cn0KLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiBJQ2xhc3NGYWN0b3J5X0xvY2tTZXJ2ZXIKICovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJQ2xhc3NGYWN0b3J5X2ZuTG9ja1NlcnZlcihMUENMQVNTRkFDVE9SWSBpZmFjZSwgQk9PTCBmTG9jaykKewoJSUNPTV9USElTKElDbGFzc0ZhY3RvcnlJbXBsLGlmYWNlKTsKCVRSQUNFKCIlcC0+KDB4JXgpLCBub3QgaW1wbGVtZW50ZWRcbiIsVGhpcywgZkxvY2spOwoJcmV0dXJuIEVfTk9USU1QTDsKfQoKc3RhdGljIElDT01fVlRBQkxFKElDbGFzc0ZhY3RvcnkpIGNsZnZ0ID0gCnsKICAgIElDT01fTVNWVEFCTEVfQ09NUEFUX0R1bW15UlRUSVZBTFVFCiAgICBJQ2xhc3NGYWN0b3J5X2ZuUXVlcnlJbnRlcmZhY2UsCiAgICBJQ2xhc3NGYWN0b3J5X2ZuQWRkUmVmLAogIElDbGFzc0ZhY3RvcnlfZm5SZWxlYXNlLAogIElDbGFzc0ZhY3RvcnlfZm5DcmVhdGVJbnN0YW5jZSwKICBJQ2xhc3NGYWN0b3J5X2ZuTG9ja1NlcnZlcgp9OwoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIERlZmF1bHQgQ2xhc3NGYWN0b3J5IEltcGxlbWVudGF0aW9uCiAqCiAqIFNIQ3JlYXRlRGVmQ2xhc3NPYmplY3QKICoKICogTk9URVMKICogIGhlbHBlciBmdW5jdGlvbiBmb3IgZGxsJ3Mgd2l0aG91dCBhIG93biBjbGFzc2ZhY3RvcnkKICogIGEgZ2VuZXJpYyBjbGFzc2ZhY3RvcnkgaXMgcmV0dXJuZWQKICogIHdoZW4gdGhlIENyZWF0ZUluc3RhbmNlIG9mIHRoZSBjZiBpcyBjYWxsZWQgdGhlIGNhbGxiYWNrIGlzIGV4ZWN1dGVkCiAqLwp0eXBlZGVmIEhSRVNVTFQgKENBTExCQUNLICogTFBGTkNSRUFURUlOU1RBTkNFKShJVW5rbm93biogcFVua091dGVyLCBSRUZJSUQgcmlpZCwgTFBWT0lEKiBwcHZPYmplY3QpOwoKdHlwZWRlZiBzdHJ1Y3QKewogICAgSUNPTV9WRklFTEQoSUNsYXNzRmFjdG9yeSk7CiAgICBEV09SRCAgICAgICAgICAgICAgICAgICAgICAgcmVmOwogICAgQ0xTSUQJCQkqcmNsc2lkOwogICAgTFBGTkNSRUFURUlOU1RBTkNFCQlscGZuQ0k7CiAgICBjb25zdCBJSUQgKgkJCXJpaWRJbnN0OwogICAgVUlOVCAqCQkJcGNSZWZEbGw7IC8qIHBvaW50ZXIgdG8gcmVmY291bnRlciBpbiBleHRlcm5hbCBkbGwgKHVncnJyLi4uKSAqLwp9IElEZWZDbEZJbXBsOwoKc3RhdGljIElDT01fVlRBQkxFKElDbGFzc0ZhY3RvcnkpIGRjbGZ2dDsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgSURlZkNsRl9mbkNvbnN0cnVjdG9yCiAqLwoKSUNsYXNzRmFjdG9yeSAqIElEZWZDbEZfZm5Db25zdHJ1Y3RvcihMUEZOQ1JFQVRFSU5TVEFOQ0UgbHBmbkNJLCBVSU5UICogcGNSZWZEbGwsIFJFRklJRCByaWlkSW5zdCkKewoJSURlZkNsRkltcGwqIGxwY2xmOwoJY2hhcgl4cmlpZEluc3RbNTBdOwoKCVdJTkVfU3RyaW5nRnJvbUNMU0lEKChMUENMU0lEKXJpaWRJbnN0LHhyaWlkSW5zdCk7CgoJbHBjbGYgPSAoSURlZkNsRkltcGwqKUhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLDAsc2l6ZW9mKElEZWZDbEZJbXBsKSk7CglscGNsZi0+cmVmID0gMTsKCUlDT01fVlRCTChscGNsZikgPSAmZGNsZnZ0OwoJbHBjbGYtPmxwZm5DSSA9IGxwZm5DSTsKCWxwY2xmLT5wY1JlZkRsbCA9IHBjUmVmRGxsOwoKCWlmIChwY1JlZkRsbCkgCgkgICgqcGNSZWZEbGwpKys7CgoJbHBjbGYtPnJpaWRJbnN0ID0gcmlpZEluc3Q7CgoJVFJBQ0UoIiglcClcblx0SUlEOlx0JXNcbiIsbHBjbGYsIHhyaWlkSW5zdCk7CglzaGVsbDMyX09iakNvdW50Kys7CglyZXR1cm4gKExQQ0xBU1NGQUNUT1JZKWxwY2xmOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgSURlZkNsRl9mblF1ZXJ5SW50ZXJmYWNlCiAqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSURlZkNsRl9mblF1ZXJ5SW50ZXJmYWNlKAogIExQQ0xBU1NGQUNUT1JZIGlmYWNlLCBSRUZJSUQgcmlpZCwgTFBWT0lEICpwcHZPYmopCnsKCUlDT01fVEhJUyhJRGVmQ2xGSW1wbCxpZmFjZSk7CgljaGFyCXhyaWlkWzUwXTsKCVdJTkVfU3RyaW5nRnJvbUNMU0lEKChMUENMU0lEKXJpaWQseHJpaWQpOwoJVFJBQ0UoIiglcCktPihcblx0SUlEOlx0JXMpXG4iLFRoaXMseHJpaWQpOwoKCSpwcHZPYmogPSBOVUxMOwoKCWlmKElzRXF1YWxJSUQocmlpZCwgJklJRF9JVW5rbm93bikpICAgICAgICAgIC8qSVVua25vd24qLwoJeyAqcHB2T2JqID0gVGhpczsgCgl9CgllbHNlIGlmKElzRXF1YWxJSUQocmlpZCwgJklJRF9JQ2xhc3NGYWN0b3J5KSkgIC8qSUNsYXNzRmFjdG9yeSovCgl7ICpwcHZPYmogPSAoSUNsYXNzRmFjdG9yeSopVGhpczsKCX0gICAKCglpZigqcHB2T2JqKQoJeyBJVW5rbm93bl9BZGRSZWYoKExQVU5LTk9XTikqcHB2T2JqKTsgIAkKCSAgVFJBQ0UoIi0tIEludGVyZmFjZTogKCVwKS0+KCVwKVxuIixwcHZPYmosKnBwdk9iaik7CgkgIHJldHVybiBTX09LOwoJfQoJVFJBQ0UoIi0tIEludGVyZmFjZTogJXMgRV9OT0lOVEVSRkFDRVxuIiwgeHJpaWQpOwoJcmV0dXJuIEVfTk9JTlRFUkZBQ0U7Cn0gIAovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqIElEZWZDbEZfZm5BZGRSZWYKICovCnN0YXRpYyBVTE9ORyBXSU5BUEkgSURlZkNsRl9mbkFkZFJlZihMUENMQVNTRkFDVE9SWSBpZmFjZSkKewoJSUNPTV9USElTKElEZWZDbEZJbXBsLGlmYWNlKTsKCVRSQUNFKCIoJXApLT4oY291bnQ9JWx1KVxuIixUaGlzLFRoaXMtPnJlZik7CgoJc2hlbGwzMl9PYmpDb3VudCsrOwoKCXJldHVybiArKyhUaGlzLT5yZWYpOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURlZkNsRl9mblJlbGVhc2UKICovCnN0YXRpYyBVTE9ORyBXSU5BUEkgSURlZkNsRl9mblJlbGVhc2UoTFBDTEFTU0ZBQ1RPUlkgaWZhY2UpCnsKCUlDT01fVEhJUyhJRGVmQ2xGSW1wbCxpZmFjZSk7CglUUkFDRSgiKCVwKS0+KGNvdW50PSVsdSlcbiIsVGhpcyxUaGlzLT5yZWYpOwoKCXNoZWxsMzJfT2JqQ291bnQtLTsKCglpZiAoIS0tKFRoaXMtPnJlZikpIAoJeyAKCSAgaWYgKFRoaXMtPnBjUmVmRGxsKSAKCSAgICAoKlRoaXMtPnBjUmVmRGxsKS0tOwoKCSAgVFJBQ0UoIi0tIGRlc3Ryb3lpbmcgSUNsYXNzRmFjdG9yeSglcClcbiIsVGhpcyk7CgkgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksMCxUaGlzKTsKCSAgcmV0dXJuIDA7Cgl9CglyZXR1cm4gVGhpcy0+cmVmOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURlZkNsRl9mbkNyZWF0ZUluc3RhbmNlCiAqLwpzdGF0aWMgSFJFU1VMVCBXSU5BUEkgSURlZkNsRl9mbkNyZWF0ZUluc3RhbmNlKAogIExQQ0xBU1NGQUNUT1JZIGlmYWNlLCBMUFVOS05PV04gcFVua091dGVyLCBSRUZJSUQgcmlpZCwgTFBWT0lEICpwcHZPYmplY3QpCnsKCUlDT01fVEhJUyhJRGVmQ2xGSW1wbCxpZmFjZSk7CgljaGFyCXhyaWlkWzUwXTsKCglXSU5FX1N0cmluZ0Zyb21DTFNJRCgoTFBDTFNJRClyaWlkLHhyaWlkKTsKCVRSQUNFKCIlcC0+KCVwLFxuXHRJSUQ6XHQlcywlcClcbiIsVGhpcyxwVW5rT3V0ZXIseHJpaWQscHB2T2JqZWN0KTsKCgkqcHB2T2JqZWN0ID0gTlVMTDsKCQkKCWlmKHBVbmtPdXRlcikKCSAgcmV0dXJuKENMQVNTX0VfTk9BR0dSRUdBVElPTik7CgoJaWYgKCBUaGlzLT5yaWlkSW5zdD09TlVMTCB8fAoJICAgICBJc0VxdWFsQ0xTSUQocmlpZCwgVGhpcy0+cmlpZEluc3QpIHx8CgkgICAgIElzRXF1YWxDTFNJRChyaWlkLCAmSUlEX0lVbmtub3duKSApCgl7CgkgIHJldHVybiBUaGlzLT5scGZuQ0kocFVua091dGVyLCByaWlkLCBwcHZPYmplY3QpOwoJfQoKCUVSUigidW5rbm93biBJSUQgcmVxdWVzdGVkXG5cdElJRDpcdCVzXG4iLHhyaWlkKTsKCXJldHVybiBFX05PSU5URVJGQUNFOwp9Ci8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogSURlZkNsRl9mbkxvY2tTZXJ2ZXIKICovCnN0YXRpYyBIUkVTVUxUIFdJTkFQSSBJRGVmQ2xGX2ZuTG9ja1NlcnZlcihMUENMQVNTRkFDVE9SWSBpZmFjZSwgQk9PTCBmTG9jaykKewoJSUNPTV9USElTKElEZWZDbEZJbXBsLGlmYWNlKTsKCVRSQUNFKCIlcC0+KDB4JXgpLCBub3QgaW1wbGVtZW50ZWRcbiIsVGhpcywgZkxvY2spOwoJcmV0dXJuIEVfTk9USU1QTDsKfQoKc3RhdGljIElDT01fVlRBQkxFKElDbGFzc0ZhY3RvcnkpIGRjbGZ2dCA9IAp7CiAgICBJQ09NX01TVlRBQkxFX0NPTVBBVF9EdW1teVJUVElWQUxVRQogICAgSURlZkNsRl9mblF1ZXJ5SW50ZXJmYWNlLAogICAgSURlZkNsRl9mbkFkZFJlZiwKICBJRGVmQ2xGX2ZuUmVsZWFzZSwKICBJRGVmQ2xGX2ZuQ3JlYXRlSW5zdGFuY2UsCiAgSURlZkNsRl9mbkxvY2tTZXJ2ZXIKfTsKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogU0hDcmVhdGVEZWZDbGFzc09iamVjdAkJCVtTSEVMTDMyLjcwXQogKi8KSFJFU1VMVCBXSU5BUEkgU0hDcmVhdGVEZWZDbGFzc09iamVjdCgKCVJFRklJRAlyaWlkLAkJCQkKCUxQVk9JRCoJcHB2LAkKCUxQRk5DUkVBVEVJTlNUQU5DRSBscGZuQ0ksCS8qIGNyZWF0ZSBpbnN0YW5jZSBjYWxsYmFjayBlbnRyeSAqLwoJVUlOVAkqcGNSZWZEbGwsCQkvKiByZWYgY291bnQgb2YgdGhlIGRsbCAqLwoJUkVGSUlECXJpaWRJbnN0KQkJLyogb3B0aW9uYWwgaW50ZXJmYWNlIHRvIHRoZSBpbnN0YW5jZSAqLwp7CgoJY2hhciB4cmlpZFs1MF0seHJpaWRJbnN0WzUwXTsKCVdJTkVfU3RyaW5nRnJvbUNMU0lEKChMUENMU0lEKXJpaWQseHJpaWQpOwoJV0lORV9TdHJpbmdGcm9tQ0xTSUQoKExQQ0xTSUQpcmlpZEluc3QseHJpaWRJbnN0KTsKCglUUkFDRSgiXG5cdElJRDpcdCVzICVwICVwICVwIFxuXHRJSURJbnM6XHQlc1xuIiwKCXhyaWlkLCBwcHYsIGxwZm5DSSwgcGNSZWZEbGwsIHhyaWlkSW5zdCk7CgoJaWYgKCBJc0VxdWFsQ0xTSUQocmlpZCwgJklJRF9JQ2xhc3NGYWN0b3J5KSApCgl7CgkgIElDbGFzc0ZhY3RvcnkgKiBwY2YgPSBJRGVmQ2xGX2ZuQ29uc3RydWN0b3IobHBmbkNJLCBwY1JlZkRsbCwgcmlpZEluc3QpOwoJICBpZiAocGNmKQoJICB7CgkgICAgKnBwdiA9IHBjZjsKCSAgICByZXR1cm4gTk9FUlJPUjsKCSAgfQoJICByZXR1cm4gRV9PVVRPRk1FTU9SWTsKCX0KCXJldHVybiBFX05PSU5URVJGQUNFOwp9Cgo=