LyoKICogTG9jYWxlIHN1cHBvcnQKICoKICogQ29weXJpZ2h0IDE5OTUgTWFydGluIHZvbiBMb2V3aXMKICogQ29weXJpZ2h0IDE5OTggRGF2aWQgTGVlIExhbWJlcnQKICogQ29weXJpZ2h0IDIwMDAgSnVsaW8gQ+lzYXIgR+F6cXVlegogKiBDb3B5cmlnaHQgMjAwMiBBbGV4YW5kcmUgSnVsbGlhcmQgZm9yIENvZGVXZWF2ZXJzCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKICogbW9kaWZ5IGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIKICogdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlIEdOVQogKiBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYWxvbmcgd2l0aCB0aGlzIGxpYnJhcnk7IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKICogRm91bmRhdGlvbiwgSW5jLiwgNTkgVGVtcGxlIFBsYWNlLCBTdWl0ZSAzMzAsIEJvc3RvbiwgTUEgIDAyMTExLTEzMDcgIFVTQQogKi8KCiNpbmNsdWRlICJjb25maWcuaCIKI2luY2x1ZGUgIndpbmUvcG9ydC5oIgoKI2luY2x1ZGUgPGFzc2VydC5oPgojaW5jbHVkZSA8c3RyaW5nLmg+CiNpbmNsdWRlIDxzdGRhcmcuaD4KI2luY2x1ZGUgPHN0ZGlvLmg+CiNpbmNsdWRlIDxjdHlwZS5oPgojaW5jbHVkZSA8c3RkbGliLmg+CgojaW5jbHVkZSAibnRzdGF0dXMuaCIKI2RlZmluZSBXSU4zMl9OT19TVEFUVVMKI2luY2x1ZGUgIndpbmRlZi5oIgojaW5jbHVkZSAid2luYmFzZS5oIgojaW5jbHVkZSAid2ludXNlci5oIiAgLyogZm9yIFJUX1NUUklOR1cgKi8KI2luY2x1ZGUgIndpbnRlcm5sLmgiCiNpbmNsdWRlICJ3aW5lL3VuaWNvZGUuaCIKI2luY2x1ZGUgIndpbm5scy5oIgojaW5jbHVkZSAid2luZXJyb3IuaCIKI2luY2x1ZGUgIndpbnZlci5oIgojaW5jbHVkZSAia2VybmVsX3ByaXZhdGUuaCIKI2luY2x1ZGUgIndpbmUvZGVidWcuaCIKCldJTkVfREVGQVVMVF9ERUJVR19DSEFOTkVMKG5scyk7CgojZGVmaW5lIExPQ0FMRV9MT0NBTEVJTkZPRkxBR1NNQVNLIChMT0NBTEVfTk9VU0VST1ZFUlJJREV8TE9DQUxFX1VTRV9DUF9BQ1B8TE9DQUxFX1JFVFVSTl9OVU1CRVIpCgovKiBjdXJyZW50IGNvZGUgcGFnZXMgKi8Kc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKmFuc2lfY3B0YWJsZTsKc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKm9lbV9jcHRhYmxlOwpzdGF0aWMgY29uc3QgdW5pb24gY3B0YWJsZSAqbWFjX2NwdGFibGU7CnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICp1bml4X2NwdGFibGU7ICAvKiBOVUxMIGlmIFVURjggKi8KCnN0YXRpYyBIQU5ETEUgTkxTX1JlZ09wZW5LZXkoSEFORExFIGhSb290S2V5LCBMUENXU1RSIHN6S2V5TmFtZSk7CnN0YXRpYyBIQU5ETEUgTkxTX1JlZ09wZW5TdWJLZXkoSEFORExFIGhSb290S2V5LCBMUENXU1RSIHN6S2V5TmFtZSk7CgpzdGF0aWMgY29uc3QgV0NIQVIgc3pObHNLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1wwJwp9OwoKLyogQ2hhcnNldCB0byBjb2RlcGFnZSBtYXAsIHNvcnRlZCBieSBuYW1lLiAqLwpzdGF0aWMgY29uc3Qgc3RydWN0IGNoYXJzZXRfZW50cnkKewogICAgY29uc3QgY2hhciAqY2hhcnNldF9uYW1lOwogICAgVUlOVCAgICAgICAgY29kZXBhZ2U7Cn0gY2hhcnNldF9uYW1lc1tdID0KewogICAgeyAiQklHNSIsIDk1MCB9LAogICAgeyAiQ1AxMjUwIiwgMTI1MCB9LAogICAgeyAiQ1AxMjUxIiwgMTI1MSB9LAogICAgeyAiQ1AxMjUyIiwgMTI1MiB9LAogICAgeyAiQ1AxMjUzIiwgMTI1MyB9LAogICAgeyAiQ1AxMjU0IiwgMTI1NCB9LAogICAgeyAiQ1AxMjU1IiwgMTI1NSB9LAogICAgeyAiQ1AxMjU2IiwgMTI1NiB9LAogICAgeyAiQ1AxMjU3IiwgMTI1NyB9LAogICAgeyAiQ1AxMjU4IiwgMTI1OCB9LAogICAgeyAiQ1A5MzIiLCA5MzIgfSwKICAgIHsgIkNQOTM2IiwgOTM2IH0sCiAgICB7ICJDUDk0OSIsIDk0OSB9LAogICAgeyAiQ1A5NTAiLCA5NTAgfSwKICAgIHsgIkVVQ0pQIiwgMjA5MzIgfSwKICAgIHsgIkdCMjMxMiIsIDkzNiB9LAogICAgeyAiSUJNMDM3IiwgMzcgfSwKICAgIHsgIklCTTEwMjYiLCAxMDI2IH0sCiAgICB7ICJJQk00MjQiLCA0MjQgfSwKICAgIHsgIklCTTQzNyIsIDQzNyB9LAogICAgeyAiSUJNNTAwIiwgNTAwIH0sCiAgICB7ICJJQk04NTAiLCA4NTAgfSwKICAgIHsgIklCTTg1MiIsIDg1MiB9LAogICAgeyAiSUJNODU1IiwgODU1IH0sCiAgICB7ICJJQk04NTciLCA4NTcgfSwKICAgIHsgIklCTTg2MCIsIDg2MCB9LAogICAgeyAiSUJNODYxIiwgODYxIH0sCiAgICB7ICJJQk04NjIiLCA4NjIgfSwKICAgIHsgIklCTTg2MyIsIDg2MyB9LAogICAgeyAiSUJNODY0IiwgODY0IH0sCiAgICB7ICJJQk04NjUiLCA4NjUgfSwKICAgIHsgIklCTTg2NiIsIDg2NiB9LAogICAgeyAiSUJNODY5IiwgODY5IH0sCiAgICB7ICJJQk04NzQiLCA4NzQgfSwKICAgIHsgIklCTTg3NSIsIDg3NSB9LAogICAgeyAiSVNPODg1OTEiLCAyODU5MSB9LAogICAgeyAiSVNPODg1OTEwIiwgMjg2MDAgfSwKICAgIHsgIklTTzg4NTkxMyIsIDI4NjAzIH0sCiAgICB7ICJJU084ODU5MTQiLCAyODYwNCB9LAogICAgeyAiSVNPODg1OTE1IiwgMjg2MDUgfSwKICAgIHsgIklTTzg4NTkxNiIsIDI4NjA2IH0sCiAgICB7ICJJU084ODU5MiIsIDI4NTkyIH0sCiAgICB7ICJJU084ODU5MyIsIDI4NTkzIH0sCiAgICB7ICJJU084ODU5NCIsIDI4NTk0IH0sCiAgICB7ICJJU084ODU5NSIsIDI4NTk1IH0sCiAgICB7ICJJU084ODU5NiIsIDI4NTk2IH0sCiAgICB7ICJJU084ODU5NyIsIDI4NTk3IH0sCiAgICB7ICJJU084ODU5OCIsIDI4NTk4IH0sCiAgICB7ICJJU084ODU5OSIsIDI4NTk5IH0sCiAgICB7ICJLT0k4UiIsIDIwODY2IH0sCiAgICB7ICJLT0k4VSIsIDIxODY2IH0sCiAgICB7ICJVVEY4IiwgQ1BfVVRGOCB9Cn07CgojZGVmaW5lIE5MU19NQVhfTEFOR1VBR0VTIDIwCnR5cGVkZWYgc3RydWN0IHsKICAgIFdDSEFSIGxhbmdbMTI4XTsKICAgIFdDSEFSIGNvdW50cnlbNF07CiAgICBMQU5HSUQgZm91bmRfbGFuZ19pZFtOTFNfTUFYX0xBTkdVQUdFU107CiAgICBpbnQgbl9mb3VuZDsKfSBMQU5HX0ZJTkRfREFUQTsKCgovKiBjb3B5IFVuaWNvZGUgc3RyaW5nIHRvIEFzY2lpIHdpdGhvdXQgdXNpbmcgY29kZXBhZ2VzICovCnN0YXRpYyBpbmxpbmUgdm9pZCBzdHJjcHlXdG9BKCBjaGFyICpkc3QsIGNvbnN0IFdDSEFSICpzcmMgKQp7CiAgICB3aGlsZSAoKCpkc3QrKyA9ICpzcmMrKykpOwp9CgovKiBDb3B5IEFzY2lpIHN0cmluZyB0byBVbmljb2RlIHdpdGhvdXQgdXNpbmcgY29kZXBhZ2VzICovCnN0YXRpYyBpbmxpbmUgdm9pZCBzdHJjcHluQXRvVyggV0NIQVIgKmRzdCwgY29uc3QgY2hhciAqc3JjLCBzaXplX3QgbiApCnsKICAgIHdoaWxlIChuID4gMSAmJiAqc3JjKQogICAgewogICAgICAgICpkc3QrKyA9ICh1bnNpZ25lZCBjaGFyKSpzcmMrKzsKICAgICAgICBuLS07CiAgICB9CiAgICBpZiAobikgKmRzdCA9IDA7Cn0KCi8qIHJldHVybiBhIHByaW50YWJsZSBzdHJpbmcgZm9yIGEgbGFuZ3VhZ2UgaWQgKi8Kc3RhdGljIGNvbnN0IGNoYXIgKmRlYnVnc3RyX2xhbmcoIExBTkdJRCBsYW5nICkKewogICAgV0NIQVIgbGFuZ1dbNF0sIGNvdW50cnlXWzRdOwogICAgY2hhciBidWZmZXJbOF07CiAgICBMQ0lEIGxjaWQgPSBNQUtFTENJRCggbGFuZywgU09SVF9ERUZBVUxUICk7CgogICAgR2V0TG9jYWxlSW5mb1cobGNpZCwgTE9DQUxFX1NJU082MzlMQU5HTkFNRXxMT0NBTEVfTk9VU0VST1ZFUlJJREUsIGxhbmdXLCBzaXplb2YobGFuZ1cpL3NpemVvZihXQ0hBUikpOwogICAgR2V0TG9jYWxlSW5mb1cobGNpZCwgTE9DQUxFX1NJU08zMTY2Q1RSWU5BTUV8TE9DQUxFX05PVVNFUk9WRVJSSURFLCBjb3VudHJ5Vywgc2l6ZW9mKGNvdW50cnlXKS9zaXplb2YoV0NIQVIpKTsKICAgIHN0cmNweVd0b0EoIGJ1ZmZlciwgbGFuZ1cgKTsKICAgIHN0cmNhdCggYnVmZmVyLCAiXyIgKTsKICAgIHN0cmNweVd0b0EoIGJ1ZmZlciArIHN0cmxlbihidWZmZXIpLCBjb3VudHJ5VyApOwogICAgcmV0dXJuIHdpbmVfZGJnX3NwcmludGYoICIlcyIsIGJ1ZmZlciApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWdldF9sY2lkX2NvZGVwYWdlCiAqCiAqIFJldHJpZXZlIHRoZSBBTlNJIGNvZGVwYWdlIGZvciBhIGdpdmVuIGxvY2FsZS4KICovCmlubGluZSBzdGF0aWMgVUlOVCBnZXRfbGNpZF9jb2RlcGFnZSggTENJRCBsY2lkICkKewogICAgVUlOVCByZXQ7CiAgICBpZiAoIUdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfSURFRkFVTFRBTlNJQ09ERVBBR0V8TE9DQUxFX1JFVFVSTl9OVU1CRVIsIChXQ0hBUiAqKSZyZXQsCiAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YocmV0KS9zaXplb2YoV0NIQVIpICkpIHJldCA9IDA7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfY29kZXBhZ2VfdGFibGUKICoKICogRmluZCB0aGUgdGFibGUgZm9yIGEgZ2l2ZW4gY29kZXBhZ2UsIGhhbmRsaW5nIENQX0FDUCBldGMuIHBzZXVkby1jb2RlcGFnZXMKICovCnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICpnZXRfY29kZXBhZ2VfdGFibGUoIHVuc2lnbmVkIGludCBjb2RlcGFnZSApCnsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnJldCA9IE5VTEw7CgogICAgYXNzZXJ0KCBhbnNpX2NwdGFibGUgKTsgIC8qIGluaXQgbXVzdCBoYXZlIGJlZW4gZG9uZSBhbHJlYWR5ICovCgogICAgc3dpdGNoKGNvZGVwYWdlKQogICAgewogICAgY2FzZSBDUF9BQ1A6CiAgICAgICAgcmV0dXJuIGFuc2lfY3B0YWJsZTsKICAgIGNhc2UgQ1BfT0VNQ1A6CiAgICAgICAgcmV0dXJuIG9lbV9jcHRhYmxlOwogICAgY2FzZSBDUF9NQUNDUDoKICAgICAgICByZXR1cm4gbWFjX2NwdGFibGU7CiAgICBjYXNlIENQX1VURjc6CiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENQX1RIUkVBRF9BQ1A6CiAgICAgICAgaWYgKCEoY29kZXBhZ2UgPSBrZXJuZWxfZ2V0X3RocmVhZF9kYXRhKCktPmNvZGVfcGFnZSkpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBkZWZhdWx0OgogICAgICAgIGlmIChjb2RlcGFnZSA9PSBhbnNpX2NwdGFibGUtPmluZm8uY29kZXBhZ2UpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG9lbV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gb2VtX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG1hY19jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gbWFjX2NwdGFibGU7CiAgICAgICAgcmV0ID0gd2luZV9jcF9nZXRfdGFibGUoIGNvZGVwYWdlICk7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWNyZWF0ZV9yZWdpc3RyeV9rZXkKICoKICogQ3JlYXRlIHRoZSBDb250cm9sIFBhbmVsXFxJbnRlcm5hdGlvbmFsIHJlZ2lzdHJ5IGtleS4KICovCmlubGluZSBzdGF0aWMgSEFORExFIGNyZWF0ZV9yZWdpc3RyeV9rZXkodm9pZCkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGludGxXW10gPSB7J0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCcgJywnUCcsJ2EnLCduJywnZScsJ2wnLCdcXCcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAnSScsJ24nLCd0JywnZScsJ3InLCduJywnYScsJ3QnLCdpJywnbycsJ24nLCdhJywnbCcsMH07CiAgICBPQkpFQ1RfQVRUUklCVVRFUyBhdHRyOwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVc7CiAgICBIQU5ETEUgaGtleTsKCiAgICBpZiAoUnRsT3BlbkN1cnJlbnRVc2VyKCBLRVlfQUxMX0FDQ0VTUywgJmhrZXkgKSAhPSBTVEFUVVNfU1VDQ0VTUykgcmV0dXJuIDA7CgogICAgYXR0ci5MZW5ndGggPSBzaXplb2YoYXR0cik7CiAgICBhdHRyLlJvb3REaXJlY3RvcnkgPSBoa2V5OwogICAgYXR0ci5PYmplY3ROYW1lID0gJm5hbWVXOwogICAgYXR0ci5BdHRyaWJ1dGVzID0gMDsKICAgIGF0dHIuU2VjdXJpdHlEZXNjcmlwdG9yID0gTlVMTDsKICAgIGF0dHIuU2VjdXJpdHlRdWFsaXR5T2ZTZXJ2aWNlID0gTlVMTDsKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmbmFtZVcsIGludGxXICk7CgogICAgaWYgKE50Q3JlYXRlS2V5KCAmaGtleSwgS0VZX0FMTF9BQ0NFU1MsICZhdHRyLCAwLCBOVUxMLCAwLCBOVUxMICkgIT0gU1RBVFVTX1NVQ0NFU1MpIGhrZXkgPSAwOwogICAgTnRDbG9zZSggYXR0ci5Sb290RGlyZWN0b3J5ICk7CiAgICByZXR1cm4gaGtleTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJTE9DQUxFX0luaXRSZWdpc3RyeQogKgogKiBVcGRhdGUgcmVnaXN0cnkgY29udGVudHMgb24gc3RhcnR1cCBpZiB0aGUgdXNlciBsb2NhbGUgaGFzIGNoYW5nZWQuCiAqIFRoaXMgc2ltdWxhdGVzIHRoZSBhY3Rpb24gb2YgdGhlIFdpbmRvd3MgY29udHJvbCBwYW5lbC4KICovCnZvaWQgTE9DQUxFX0luaXRSZWdpc3RyeSh2b2lkKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgQ29kZXBhZ2VXW10gPSB7J0MnLCdvJywnZCcsJ2UnLCdwJywnYScsJ2cnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBhY3BXW10gPSB7J0EnLCdDJywnUCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgb2VtY3BXW10gPSB7J08nLCdFJywnTScsJ0MnLCdQJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBtYWNjcFdbXSA9IHsnTScsJ0EnLCdDJywnQycsJ1AnLDB9OwogICAgc3RhdGljIGNvbnN0IHN0cnVjdAogICAgewogICAgICAgIExQQ1dTVFIgbmFtZTsKICAgICAgICBVU0hPUlQgdmFsdWU7CiAgICB9IHVwZGF0ZV9jcF92YWx1ZXNbXSA9IHsKICAgICAgICB7IGFjcFcsIExPQ0FMRV9JREVGQVVMVEFOU0lDT0RFUEFHRSB9LAogICAgICAgIHsgb2VtY3BXLCBMT0NBTEVfSURFRkFVTFRDT0RFUEFHRSB9LAogICAgICAgIHsgbWFjY3BXLCBMT0NBTEVfSURFRkFVTFRNQUNDT0RFUEFHRSB9CiAgICB9OwogICAgc3RhdGljIGNvbnN0IFVTSE9SVCB1cGRhdGVWYWx1ZXNbXSA9IHsKICAgICAgTE9DQUxFX1NMQU5HVUFHRSwKICAgICAgTE9DQUxFX1NDT1VOVFJZLCBMT0NBTEVfSUNPVU5UUlksCiAgICAgIExPQ0FMRV9TMTE1OSwgTE9DQUxFX1MyMzU5LAogICAgICBMT0NBTEVfU1RJTUUsIExPQ0FMRV9JVElNRSwKICAgICAgTE9DQUxFX0lUTFpFUk8sCiAgICAgIExPQ0FMRV9TU0hPUlREQVRFLAogICAgICBMT0NBTEVfU0xPTkdEQVRFLAogICAgICBMT0NBTEVfU0RBVEUsCiAgICAgIExPQ0FMRV9TQ1VSUkVOQ1ksIExPQ0FMRV9JQ1VSUkVOQ1ksCiAgICAgIExPQ0FMRV9JTkVHQ1VSUiwKICAgICAgTE9DQUxFX0lDVVJSRElHSVRTLAogICAgICBMT0NBTEVfU0RFQ0lNQUwsCiAgICAgIExPQ0FMRV9TTElTVCwKICAgICAgTE9DQUxFX1NUSE9VU0FORCwKICAgICAgTE9DQUxFX0lESUdJVFMsCiAgICAgIExPQ0FMRV9JRElHSVRTVUJTVElUVVRJT04sCiAgICAgIExPQ0FMRV9TTkFUSVZFRElHSVRTLAogICAgICBMT0NBTEVfSVRJTUVNQVJLUE9TTiwKICAgICAgTE9DQUxFX0lDQUxFTkRBUlRZUEUsCiAgICAgIExPQ0FMRV9JTFpFUk8sCiAgICAgIExPQ0FMRV9JTUVBU1VSRQogICAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBMb2NhbGVXW10gPSB7J0wnLCdvJywnYycsJ2EnLCdsJywnZScsMH07CiAgICBVTklDT0RFX1NUUklORyBuYW1lVzsKICAgIGNoYXIgYnVmZmVyWzIwXTsKICAgIFdDSEFSIGJ1ZmZlcldbODBdOwogICAgRFdPUkQgY291bnQsIGk7CiAgICBIQU5ETEUgaGtleTsKICAgIExDSUQgbGNpZCA9IEdldFVzZXJEZWZhdWx0TENJRCgpOwoKICAgIGlmICghKGhrZXkgPSBjcmVhdGVfcmVnaXN0cnlfa2V5KCkpKQogICAgICAgIHJldHVybjsgIC8qIGRvbid0IGRvIGFueXRoaW5nIGlmIHdlIGNhbid0IGNyZWF0ZSB0aGUgcmVnaXN0cnkga2V5ICovCgogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgTG9jYWxlVyApOwogICAgY291bnQgPSBzaXplb2YoYnVmZmVyVyk7CiAgICBpZiAoIU50UXVlcnlWYWx1ZUtleShoa2V5LCAmbmFtZVcsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLCAoTFBCWVRFKWJ1ZmZlclcsIGNvdW50LCAmY291bnQpKQogICAgewogICAgICAgIGNvbnN0IEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICppbmZvID0gKEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICopYnVmZmVyVzsKICAgICAgICBMUENXU1RSIHN6VmFsdWVUZXh0ID0gKExQQ1dTVFIpaW5mby0+RGF0YTsKCiAgICAgICAgaWYgKHN0cnRvdWxXKCBzelZhbHVlVGV4dCwgTlVMTCwgMTYgKSA9PSBsY2lkKSAgLyogYWxyZWFkeSBzZXQgY29ycmVjdGx5ICovCiAgICAgICAgewogICAgICAgICAgICBOdENsb3NlKCBoa2V5ICk7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CiAgICAgICAgVFJBQ0UoICJ1cGRhdGluZyByZWdpc3RyeSwgbG9jYWxlIGNoYW5nZWQgJXMgLT4gJTA4bHhcbiIsIGRlYnVnc3RyX3coc3pWYWx1ZVRleHQpLCBsY2lkICk7CiAgICB9CiAgICBlbHNlIFRSQUNFKCAidXBkYXRpbmcgcmVnaXN0cnksIGxvY2FsZSBjaGFuZ2VkIG5vbmUgLT4gJTA4bHhcbiIsIGxjaWQgKTsKCiAgICBzcHJpbnRmKCBidWZmZXIsICIlMDhseCIsIGxjaWQgKTsKICAgIC8qIE5vdGU6ICc5JyBjb25zdGFudCBiZWxvdyBpcyBzdHJsZW4oYnVmZmVyKSArIDEgKi8KICAgIFJ0bE11bHRpQnl0ZVRvVW5pY29kZU4oIGJ1ZmZlclcsIHNpemVvZihidWZmZXJXKSwgTlVMTCwgYnVmZmVyLCA5ICk7CiAgICBOdFNldFZhbHVlS2V5KCBoa2V5LCAmbmFtZVcsIDAsIFJFR19TWiwgYnVmZmVyVywgOSAqIHNpemVvZihXQ0hBUikgKTsKICAgIE50Q2xvc2UoIGhrZXkgKTsKCiAgICBmb3IgKGkgPSAwOyBpIDwgc2l6ZW9mKHVwZGF0ZVZhbHVlcykvc2l6ZW9mKHVwZGF0ZVZhbHVlc1swXSk7IGkrKykKICAgIHsKICAgICAgICBHZXRMb2NhbGVJbmZvVyggbGNpZCwgdXBkYXRlVmFsdWVzW2ldIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLCBidWZmZXJXLAogICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoYnVmZmVyVykvc2l6ZW9mKFdDSEFSKSApOwogICAgICAgIFNldExvY2FsZUluZm9XKCBsY2lkLCB1cGRhdGVWYWx1ZXNbaV0sIGJ1ZmZlclcgKTsKICAgIH0KCiAgICBoa2V5ID0gTkxTX1JlZ09wZW5TdWJLZXkoIE5MU19SZWdPcGVuS2V5KCAwLCBzek5sc0tleU5hbWUgKSwgQ29kZXBhZ2VXICk7CgogICAgZm9yIChpID0gMDsgaSA8IHNpemVvZih1cGRhdGVfY3BfdmFsdWVzKS9zaXplb2YodXBkYXRlX2NwX3ZhbHVlc1swXSk7IGkrKykKICAgIHsKICAgICAgICBjb3VudCA9IEdldExvY2FsZUluZm9XKCBsY2lkLCB1cGRhdGVfY3BfdmFsdWVzW2ldLnZhbHVlIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZmZlclcsIHNpemVvZihidWZmZXJXKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgdXBkYXRlX2NwX3ZhbHVlc1tpXS5uYW1lICk7CiAgICAgICAgTnRTZXRWYWx1ZUtleSggaGtleSwgJm5hbWVXLCAwLCBSRUdfU1osIGJ1ZmZlclcsIGNvdW50ICogc2l6ZW9mKFdDSEFSKSApOwogICAgfQoKICAgIE50Q2xvc2UoIGhrZXkgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgZmluZF9sYW5ndWFnZV9pZF9wcm9jCiAqLwpzdGF0aWMgQk9PTCBDQUxMQkFDSyBmaW5kX2xhbmd1YWdlX2lkX3Byb2MoIEhNT0RVTEUgaE1vZHVsZSwgTFBDV1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1dTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExQQVJBTSBsUGFyYW0gKQp7CiAgICBMQU5HX0ZJTkRfREFUQSAqbF9kYXRhID0gKExBTkdfRklORF9EQVRBICopbFBhcmFtOwogICAgTENJRCBsY2lkID0gTUFLRUxDSUQoTGFuZ0lELCBTT1JUX0RFRkFVTFQpOwogICAgV0NIQVIgYnVmX2xhbmd1YWdlWzEyOF07CiAgICBXQ0hBUiBidWZfY291bnRyeVsxMjhdOwogICAgV0NIQVIgYnVmX2VuX2xhbmd1YWdlWzEyOF07CgogICAgaWYoUFJJTUFSWUxBTkdJRChMYW5nSUQpID09IExBTkdfTkVVVFJBTCkKICAgICAgICByZXR1cm4gVFJVRTsgLyogY29udGludWUgc2VhcmNoICovCgogICAgYnVmX2xhbmd1YWdlWzBdID0gMDsKICAgIGJ1Zl9jb3VudHJ5WzBdID0gMDsKCiAgICBHZXRMb2NhbGVJbmZvVyhsY2lkLCBMT0NBTEVfU0lTTzYzOUxBTkdOQU1FfExPQ0FMRV9OT1VTRVJPVkVSUklERSwKICAgICAgICAgICAgICAgICAgIGJ1Zl9sYW5ndWFnZSwgc2l6ZW9mKGJ1Zl9sYW5ndWFnZSkvc2l6ZW9mKFdDSEFSKSk7CiAgICBHZXRMb2NhbGVJbmZvVyhsY2lkLCBMT0NBTEVfU0lTTzMxNjZDVFJZTkFNRXxMT0NBTEVfTk9VU0VST1ZFUlJJREUsCiAgICAgICAgICAgICAgICAgICBidWZfY291bnRyeSwgc2l6ZW9mKGJ1Zl9jb3VudHJ5KS9zaXplb2YoV0NIQVIpKTsKCiAgICBpZihsX2RhdGEtPmxhbmdbMF0gJiYgIXN0cmNtcGlXKGxfZGF0YS0+bGFuZywgYnVmX2xhbmd1YWdlKSkKICAgIHsKICAgICAgICBpZihsX2RhdGEtPmNvdW50cnlbMF0pCiAgICAgICAgewogICAgICAgICAgICBpZighc3RyY21waVcobF9kYXRhLT5jb3VudHJ5LCBidWZfY291bnRyeSkpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIGxfZGF0YS0+Zm91bmRfbGFuZ19pZFswXSA9IExhbmdJRDsKICAgICAgICAgICAgICAgIGxfZGF0YS0+bl9mb3VuZCA9IDE7CiAgICAgICAgICAgICAgICBUUkFDRSgiRm91bmQgaWQgJTA0WCBmb3IgbGFuZyAlcyBjb3VudHJ5ICVzXG4iLAogICAgICAgICAgICAgICAgICAgICAgTGFuZ0lELCBkZWJ1Z3N0cl93KGxfZGF0YS0+bGFuZyksIGRlYnVnc3RyX3cobF9kYXRhLT5jb3VudHJ5KSk7CiAgICAgICAgICAgICAgICByZXR1cm4gRkFMU0U7IC8qIHN0b3AgZW51bWVyYXRpb24gKi8KICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlIGdvdG8gZm91bmQ7IC8qIGxfZGF0YS0+Y291bnRyeSBub3Qgc3BlY2lmaWVkICovCiAgICB9CgogICAgLyogSnVzdCBpbiBjYXNlLCBjaGVjayBMT0NBTEVfU0VOR0xBTkdVQUdFIHRvbywKICAgICAqIGluIGhvcGUgdGhhdCBwb3NzaWJsZSBhbGlhcyBuYW1lIG1pZ2h0IGhhdmUgdGhhdCB2YWx1ZS4KICAgICAqLwogICAgYnVmX2VuX2xhbmd1YWdlWzBdID0gMDsKICAgIEdldExvY2FsZUluZm9XKGxjaWQsIExPQ0FMRV9TRU5HTEFOR1VBR0V8TE9DQUxFX05PVVNFUk9WRVJSSURFLAogICAgICAgICAgICAgICAgICAgYnVmX2VuX2xhbmd1YWdlLCBzaXplb2YoYnVmX2VuX2xhbmd1YWdlKS9zaXplb2YoV0NIQVIpKTsKCiAgICBpZihsX2RhdGEtPmxhbmdbMF0gJiYgIXN0cmNtcGlXKGxfZGF0YS0+bGFuZywgYnVmX2VuX2xhbmd1YWdlKSkgZ290byBmb3VuZDsKICAgIHJldHVybiBUUlVFOyAgLyogbm90IGZvdW5kLCBjb250aW51ZSBzZWFyY2ggKi8KCmZvdW5kOgogICAgbF9kYXRhLT5mb3VuZF9sYW5nX2lkW2xfZGF0YS0+bl9mb3VuZF0gPSBMYW5nSUQ7CiAgICBsX2RhdGEtPm5fZm91bmQrKzsKICAgIFRSQUNFKCJGb3VuZCBpZCAlMDRYIGZvciBsYW5nICVzXG4iLCBMYW5nSUQsIGRlYnVnc3RyX3cobF9kYXRhLT5sYW5nKSk7CiAgICByZXR1cm4gKGxfZGF0YS0+bl9mb3VuZCA8IE5MU19NQVhfTEFOR1VBR0VTKTsgLyogY29udGludWUgc2VhcmNoLCB1bmxlc3Mgd2UgaGF2ZSBlbm91Z2ggKi8KfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgZ2V0X2xhbmd1YWdlX2lkCiAqCiAqIElOUFVUOgogKglMYW5nOiBhIHN0cmluZyB3aG9zZSB0d28gZmlyc3QgY2hhcnMgYXJlIHRoZSBpc28gbmFtZSBvZiBhIGxhbmd1YWdlLgogKglDb3VudHJ5OiBhIHN0cmluZyB3aG9zZSB0d28gZmlyc3QgY2hhcnMgYXJlIHRoZSBpc28gbmFtZSBvZiBjb3VudHJ5CiAqCUNoYXJzZXQ6IGEgc3RyaW5nIGRlZmluaW5nIHRoZSBjaG9zZW4gY2hhcnNldCBlbmNvZGluZwogKglEaWFsZWN0OiBhIHN0cmluZyBkZWZpbmluZyBhIHZhcmlhdGlvbiBvZiB0aGUgbG9jYWxlCiAqCiAqCWFsbCB0aG9zZSB2YWx1ZXMgYXJlIGZyb20gdGhlIHN0YW5kYXJkaXplZCBmb3JtYXQgb2YgbG9jYWxlCiAqCW5hbWUgaW4gdW5peCB3aGljaCBpczogTGFuZ1tfQ291bnRyeV1bLkNoYXJzZXRdW0BEaWFsZWN0XQogKgogKiBSRVRVUk5TOgogKgl0aGUgbnVtZXJpYyBjb2RlIG9mIHRoZSBsYW5ndWFnZSB1c2VkIGJ5IFdpbmRvd3MKICoKICogRklYTUU6IENoYXJzZXQgYW5kIERpYWxlY3QgYXJlIG5vdCBoYW5kbGVkCiAqLwpzdGF0aWMgTEFOR0lEIGdldF9sYW5ndWFnZV9pZChMUENTVFIgTGFuZywgTFBDU1RSIENvdW50cnksIExQQ1NUUiBDaGFyc2V0LCBMUENTVFIgRGlhbGVjdCkKewogICAgTEFOR19GSU5EX0RBVEEgbF9kYXRhOwoKICAgIGlmKCFMYW5nKQogICAgewogICAgICAgIGxfZGF0YS5mb3VuZF9sYW5nX2lkWzBdID0gTUFLRUxBTkdJRChMQU5HX0VOR0xJU0gsIFNVQkxBTkdfREVGQVVMVCk7CiAgICAgICAgZ290byBFTkQ7CiAgICB9CgogICAgbF9kYXRhLm5fZm91bmQgPSAwOwogICAgc3RyY3B5bkF0b1cobF9kYXRhLmxhbmcsIExhbmcsIHNpemVvZihsX2RhdGEubGFuZykvc2l6ZW9mKFdDSEFSKSk7CgogICAgaWYgKENvdW50cnkpIHN0cmNweW5BdG9XKGxfZGF0YS5jb3VudHJ5LCBDb3VudHJ5LCBzaXplb2YobF9kYXRhLmNvdW50cnkpL3NpemVvZihXQ0hBUikpOwogICAgZWxzZSBsX2RhdGEuY291bnRyeVswXSA9IDA7CgogICAgRW51bVJlc291cmNlTGFuZ3VhZ2VzVyhrZXJuZWwzMl9oYW5kbGUsIChMUENXU1RSKVJUX1NUUklORywgKExQQ1dTVFIpTE9DQUxFX0lMQU5HVUFHRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgZmluZF9sYW5ndWFnZV9pZF9wcm9jLCAoTFBBUkFNKSZsX2RhdGEpOwoKICAgIGlmIChsX2RhdGEubl9mb3VuZCA9PSAxKSBnb3RvIEVORDsKCiAgICBpZighbF9kYXRhLm5fZm91bmQpCiAgICB7CiAgICAgICAgaWYobF9kYXRhLmNvdW50cnlbMF0pCiAgICAgICAgewogICAgICAgICAgICAvKiByZXRyeSB3aXRob3V0IGNvdW50cnkgbmFtZSAqLwogICAgICAgICAgICBsX2RhdGEuY291bnRyeVswXSA9IDA7CiAgICAgICAgICAgIEVudW1SZXNvdXJjZUxhbmd1YWdlc1coa2VybmVsMzJfaGFuZGxlLCAoTFBDV1NUUilSVF9TVFJJTkcsIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZmluZF9sYW5ndWFnZV9pZF9wcm9jLCAoTE9ORykmbF9kYXRhKTsKICAgICAgICAgICAgaWYgKCFsX2RhdGEubl9mb3VuZCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgTUVTU0FHRSgiV2FybmluZzogTGFuZ3VhZ2UgJyVzXyVzJyB3YXMgbm90IHJlY29nbml6ZWQsIGRlZmF1bHRpbmcgdG8gRW5nbGlzaC5cbiIsCiAgICAgICAgICAgICAgICAgICAgICAgIExhbmcsIENvdW50cnkpOwogICAgICAgICAgICAgICAgbF9kYXRhLmZvdW5kX2xhbmdfaWRbMF0gPSBNQUtFTEFOR0lEKExBTkdfRU5HTElTSCwgU1VCTEFOR19ERUZBVUxUKTsKICAgICAgICAgICAgfQogICAgICAgICAgICBlbHNlIE1FU1NBR0UoIldhcm5pbmc6IExhbmd1YWdlICclc18lcycgd2FzIG5vdCByZWNvZ25pemVkLCBkZWZhdWx0aW5nIHRvICclcycuXG4iLAogICAgICAgICAgICAgICAgICAgICAgICAgTGFuZywgQ291bnRyeSwgZGVidWdzdHJfbGFuZyhsX2RhdGEuZm91bmRfbGFuZ19pZFswXSkgKTsKICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgIHsKICAgICAgICAgICAgTUVTU0FHRSgiV2FybmluZzogTGFuZ3VhZ2UgJyVzJyB3YXMgbm90IHJlY29nbml6ZWQsIGRlZmF1bHRpbmcgdG8gRW5nbGlzaC5cbiIsIExhbmcpOwogICAgICAgICAgICBsX2RhdGEuZm91bmRfbGFuZ19pZFswXSA9IE1BS0VMQU5HSUQoTEFOR19FTkdMSVNILCBTVUJMQU5HX0RFRkFVTFQpOwogICAgICAgIH0KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICBpbnQgaTsKCiAgICAgICAgaWYgKENvdW50cnkgJiYgQ291bnRyeVswXSkKICAgICAgICAgICAgTUVTU0FHRSgiRm9yIGxhbmd1YWdlICclc18lcycgc2V2ZXJhbCBsYW5ndWFnZSBpZHMgd2VyZSBmb3VuZDpcbiIsIExhbmcsIENvdW50cnkpOwogICAgICAgIGVsc2UKICAgICAgICAgICAgTUVTU0FHRSgiRm9yIGxhbmd1YWdlICclcycgc2V2ZXJhbCBsYW5ndWFnZSBpZHMgd2VyZSBmb3VuZDpcbiIsIExhbmcpOwoKICAgICAgICAvKiBwcmludCBhIGxpc3Qgb2YgbGFuZ3VhZ2VzIHdpdGggdGhlaXIgZGVzY3JpcHRpb24gKi8KICAgICAgICBmb3IgKGkgPSAwOyBpIDwgbF9kYXRhLm5fZm91bmQ7IGkrKykKICAgICAgICB7CiAgICAgICAgICAgIFdDSEFSIGJ1ZmZXWzEyOF07CiAgICAgICAgICAgIGNoYXIgYnVmZkFbMTI4XTsKICAgICAgICAgICAgR2V0TG9jYWxlSW5mb1coIE1BS0VMQ0lEKCBsX2RhdGEuZm91bmRfbGFuZ19pZFtpXSwgU09SVF9ERUZBVUxUICksCiAgICAgICAgICAgICAgICAgICAgICAgICAgIExPQ0FMRV9TTEFOR1VBR0V8TE9DQUxFX05PVVNFUk9WRVJSSURFLCBidWZmVywgc2l6ZW9mKGJ1ZmZXKS9zaXplb2YoV0NIQVIpKTsKICAgICAgICAgICAgc3RyY3B5V3RvQSggYnVmZkEsIGJ1ZmZXICk7CiAgICAgICAgICAgIE1FU1NBR0UoICIgICAlcyAoJTA0WCkgLSAlc1xuIiwgZGVidWdzdHJfbGFuZyhsX2RhdGEuZm91bmRfbGFuZ19pZFtpXSksCiAgICAgICAgICAgICAgICAgICAgIGxfZGF0YS5mb3VuZF9sYW5nX2lkW2ldLCBidWZmQSApOwogICAgICAgIH0KICAgICAgICBNRVNTQUdFKCJEZWZhdWx0aW5nIHRvICclcycuIFlvdSBzaG91bGQgc3BlY2lmeSB0aGUgZXhhY3QgbGFuZ3VhZ2UgeW91IHdhbnRcbiIKICAgICAgICAgICAgICAgICJieSBkZWZpbmluZyB5b3VyIExBTkcgZW52aXJvbm1lbnQgdmFyaWFibGUgbGlrZSB0aGlzOiBMQU5HPSVzXG4iLAogICAgICAgICAgICAgICAgZGVidWdzdHJfbGFuZyhsX2RhdGEuZm91bmRfbGFuZ19pZFswXSksIGRlYnVnc3RyX2xhbmcobF9kYXRhLmZvdW5kX2xhbmdfaWRbMF0pICk7CiAgICB9CkVORDoKICAgIFRSQUNFKCJSZXR1cm5pbmcgJTA0WCAoJXMpXG4iLCBsX2RhdGEuZm91bmRfbGFuZ19pZFswXSwgZGVidWdzdHJfbGFuZyhsX2RhdGEuZm91bmRfbGFuZ19pZFswXSkpOwogICAgcmV0dXJuIGxfZGF0YS5mb3VuZF9sYW5nX2lkWzBdOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBjaGFyc2V0X2NtcCAoaW50ZXJuYWwpCiAqLwpzdGF0aWMgaW50IGNoYXJzZXRfY21wKCBjb25zdCB2b2lkICpuYW1lLCBjb25zdCB2b2lkICplbnRyeSApCnsKICAgIGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICpjaGFyc2V0ID0gKGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICopZW50cnk7CiAgICByZXR1cm4gc3RyY2FzZWNtcCggKGNvbnN0IGNoYXIgKiluYW1lLCBjaGFyc2V0LT5jaGFyc2V0X25hbWUgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfZW52X2xjaWQKICovCnN0YXRpYyBMQ0lEIGdldF9lbnZfbGNpZCggVUlOVCAqdW5peF9jcCwgY29uc3QgY2hhciAqZW52X3N0ciApCnsKICAgIGNoYXIgKmJ1ZiwgKmxhbmcsKmNvdW50cnksKmNoYXJzZXQsKmRpYWxlY3QsKm5leHQ7CiAgICBMQ0lEIHJldCA9IDA7CgogICAgaWYgKCgobGFuZyA9IGdldGVudiggIkxDX0FMTCIgKSkgJiYgKmxhbmcpIHx8CiAgICAgICAgKGVudl9zdHIgJiYgKGxhbmcgPSBnZXRlbnYoIGVudl9zdHIgKSkgJiYgKmxhbmcpIHx8CiAgICAgICAgKChsYW5nID0gZ2V0ZW52KCAiTEFORyIgKSkgJiYgKmxhbmcpKQogICAgewogICAgICAgIGlmICghc3RyY21wKGxhbmcsIlBPU0lYIikgfHwgIXN0cmNtcChsYW5nLCJDIikpIGdvdG8gZG9uZTsKCiAgICAgICAgYnVmID0gUnRsQWxsb2NhdGVIZWFwKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHJsZW4obGFuZykgKyAxICk7CiAgICAgICAgc3RyY3B5KCBidWYsIGxhbmcgKTsKICAgICAgICBsYW5nPWJ1ZjsKCiAgICAgICAgZG8gewogICAgICAgICAgICBuZXh0PXN0cmNocihsYW5nLCc6Jyk7IGlmIChuZXh0KSAqbmV4dCsrPSdcMCc7CiAgICAgICAgICAgIGRpYWxlY3Q9c3RyY2hyKGxhbmcsJ0AnKTsgaWYgKGRpYWxlY3QpICpkaWFsZWN0Kys9J1wwJzsKICAgICAgICAgICAgY2hhcnNldD1zdHJjaHIobGFuZywnLicpOyBpZiAoY2hhcnNldCkgKmNoYXJzZXQrKz0nXDAnOwogICAgICAgICAgICBjb3VudHJ5PXN0cmNocihsYW5nLCdfJyk7IGlmIChjb3VudHJ5KSAqY291bnRyeSsrPSdcMCc7CgogICAgICAgICAgICByZXQgPSBnZXRfbGFuZ3VhZ2VfaWQobGFuZywgY291bnRyeSwgY2hhcnNldCwgZGlhbGVjdCk7CiAgICAgICAgICAgIGlmIChyZXQgJiYgY2hhcnNldCAmJiB1bml4X2NwKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBjb25zdCBzdHJ1Y3QgY2hhcnNldF9lbnRyeSAqZW50cnk7CiAgICAgICAgICAgICAgICBjaGFyIGNoYXJzZXRfbmFtZVsxNl07CiAgICAgICAgICAgICAgICBzaXplX3QgaSwgajsKCiAgICAgICAgICAgICAgICAvKiByZW1vdmUgcHVuY3R1YXRpb24gY2hhcmFjdGVycyBmcm9tIGNoYXJzZXQgbmFtZSAqLwogICAgICAgICAgICAgICAgZm9yIChpID0gaiA9IDA7IGNoYXJzZXRbaV0gJiYgaiA8IHNpemVvZihjaGFyc2V0X25hbWUpLTE7IGkrKykKICAgICAgICAgICAgICAgICAgICBpZiAoaXNhbG51bShjaGFyc2V0W2ldKSkgY2hhcnNldF9uYW1lW2orK10gPSBjaGFyc2V0W2ldOwogICAgICAgICAgICAgICAgY2hhcnNldF9uYW1lW2pdID0gMDsKCiAgICAgICAgICAgICAgICBlbnRyeSA9IGJzZWFyY2goIGNoYXJzZXRfbmFtZSwgY2hhcnNldF9uYW1lcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXMpL3NpemVvZihjaGFyc2V0X25hbWVzWzBdKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXNbMF0pLCBjaGFyc2V0X2NtcCApOwogICAgICAgICAgICAgICAgaWYgKGVudHJ5KQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICp1bml4X2NwID0gZW50cnktPmNvZGVwYWdlOwogICAgICAgICAgICAgICAgICAgIFRSQUNFKCJjaGFyc2V0ICVzIHdhcyBtYXBwZWQgdG8gY3AgJXVcbiIsIGNoYXJzZXQsICp1bml4X2NwKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgICAgICBGSVhNRSgiY2hhcnNldCAlcyB3YXMgbm90IHJlY29nbml6ZWRcbiIsIGNoYXJzZXQpOwogICAgICAgICAgICB9CgogICAgICAgICAgICBsYW5nPW5leHQ7CiAgICAgICAgfSB3aGlsZSAobGFuZyAmJiAhcmV0KTsKCiAgICAgICAgaWYgKCFyZXQpIE1FU1NBR0UoIldhcm5pbmc6IGxhbmd1YWdlICclcycgbm90IHJlY29nbml6ZWQsIGRlZmF1bHRpbmcgdG8gRW5nbGlzaFxuIiwgYnVmKTsKICAgICAgICBSdGxGcmVlSGVhcCggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgYnVmICk7CiAgICB9CgogZG9uZToKICAgIGlmICghcmV0KSByZXQgPSBNQUtFTENJRCggTUFLRUxBTkdJRChMQU5HX0VOR0xJU0gsU1VCTEFOR19ERUZBVUxUKSwgU09SVF9ERUZBVUxUKSA7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRVc2VyRGVmYXVsdExhbmdJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxhbmd1YWdlIElkIGZvciB0aGUgY3VycmVudCB1c2VyLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBsYW5ndWFnZSBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICovCkxBTkdJRCBXSU5BUEkgR2V0VXNlckRlZmF1bHRMYW5nSUQodm9pZCkKewogICAgcmV0dXJuIExBTkdJREZST01MQ0lEKEdldFVzZXJEZWZhdWx0TENJRCgpKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0U3lzdGVtRGVmYXVsdExhbmdJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxhbmd1YWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBsYW5ndWFnZSBmb3IgdGhlIHN5c3RlbS4KICovCkxBTkdJRCBXSU5BUEkgR2V0U3lzdGVtRGVmYXVsdExhbmdJRCh2b2lkKQp7CiAgICByZXR1cm4gTEFOR0lERlJPTUxDSUQoR2V0U3lzdGVtRGVmYXVsdExDSUQoKSk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUdldFVzZXJEZWZhdWx0TENJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxvY2FsZSBJZCBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQ0lEIG9mIHRoZSBkZWZhdWx0IGxvY2FsZSBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICovCkxDSUQgV0lOQVBJIEdldFVzZXJEZWZhdWx0TENJRCh2b2lkKQp7CiAgICBMQ0lEIGxjaWQ7CiAgICBOdFF1ZXJ5RGVmYXVsdExvY2FsZSggVFJVRSwgJmxjaWQgKTsKICAgIHJldHVybiBsY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRTeXN0ZW1EZWZhdWx0TENJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxvY2FsZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQ0lEIG9mIHRoZSBkZWZhdWx0IGxvY2FsZSBmb3IgdGhlIHN5c3RlbS4KICovCkxDSUQgV0lOQVBJIEdldFN5c3RlbURlZmF1bHRMQ0lEKHZvaWQpCnsKICAgIExDSUQgbGNpZDsKICAgIE50UXVlcnlEZWZhdWx0TG9jYWxlKCBGQUxTRSwgJmxjaWQgKTsKICAgIHJldHVybiBsY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRVc2VyRGVmYXVsdFVJTGFuZ3VhZ2UgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCB1c2VyIGludGVyZmFjZSBsYW5ndWFnZSBJZCBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQU5HSUQgb2YgdGhlIGRlZmF1bHQgVUkgbGFuZ3VhZ2UgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqLwpMQU5HSUQgV0lOQVBJIEdldFVzZXJEZWZhdWx0VUlMYW5ndWFnZSh2b2lkKQp7CiAgICBMQU5HSUQgbGFuZzsKICAgIE50UXVlcnlEZWZhdWx0VUlMYW5ndWFnZSggJmxhbmcgKTsKICAgIHJldHVybiBsYW5nOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRTeXN0ZW1EZWZhdWx0VUlMYW5ndWFnZSAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IHVzZXIgaW50ZXJmYWNlIGxhbmd1YWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBVSSBsYW5ndWFnZSBmb3IgdGhlIHN5c3RlbS4gVGhpcyBpcwogKiAgdHlwaWNhbGx5IHRoZSBzYW1lIGxhbmd1YWdlIHVzZWQgZHVyaW5nIHRoZSBpbnN0YWxsYXRpb24gcHJvY2Vzcy4KICovCkxBTkdJRCBXSU5BUEkgR2V0U3lzdGVtRGVmYXVsdFVJTGFuZ3VhZ2Uodm9pZCkKewogICAgTEFOR0lEIGxhbmc7CiAgICBOdFF1ZXJ5SW5zdGFsbFVJTGFuZ3VhZ2UoICZsYW5nICk7CiAgICByZXR1cm4gbGFuZzsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWdldF9sb2NhbGVfdmFsdWVfbmFtZQogKgogKiBHZXRzIHRoZSByZWdpc3RyeSB2YWx1ZSBuYW1lIGZvciBhIGdpdmVuIGxjdHlwZS4KICovCnN0YXRpYyBjb25zdCBXQ0hBUiAqZ2V0X2xvY2FsZV92YWx1ZV9uYW1lKCBEV09SRCBsY3R5cGUgKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUNhbGVuZGFyVHlwZVdbXSA9IHsnaScsJ0MnLCdhJywnbCcsJ2UnLCduJywnZCcsJ2EnLCdyJywnVCcsJ3knLCdwJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUNvdW50cnlXW10gPSB7J2knLCdDJywnbycsJ3UnLCduJywndCcsJ3InLCd5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpQ3VyckRpZ2l0c1dbXSA9IHsnaScsJ0MnLCd1JywncicsJ3InLCdEJywnaScsJ2cnLCdpJywndCcsJ3MnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlDdXJyZW5jeVdbXSA9IHsnaScsJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ2MnLCd5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRGF0ZVdbXSA9IHsnaScsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlEaWdpdHNXW10gPSB7J2knLCdEJywnaScsJ2cnLCdpJywndCcsJ3MnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlGaXJzdERheU9mV2Vla1dbXSA9IHsnaScsJ0YnLCdpJywncicsJ3MnLCd0JywnRCcsJ2EnLCd5JywnTycsJ2YnLCdXJywnZScsJ2UnLCdrJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRmlyc3RXZWVrT2ZZZWFyV1tdID0geydpJywnRicsJ2knLCdyJywncycsJ3QnLCdXJywnZScsJ2UnLCdrJywnTycsJ2YnLCdZJywnZScsJ2EnLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpTERhdGVXW10gPSB7J2knLCdMJywnRCcsJ2EnLCd0JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUxaZXJvV1tdID0geydpJywnTCcsJ1onLCdlJywncicsJ28nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlNZWFzdXJlV1tdID0geydpJywnTScsJ2UnLCdhJywncycsJ3UnLCdyJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaU5lZ0N1cnJXW10gPSB7J2knLCdOJywnZScsJ2cnLCdDJywndScsJ3InLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpTmVnTnVtYmVyV1tdID0geydpJywnTicsJ2UnLCdnJywnTicsJ3UnLCdtJywnYicsJ2UnLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpUGFwZXJTaXplV1tdID0geydpJywnUCcsJ2EnLCdwJywnZScsJ3InLCdTJywnaScsJ3onLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpVExaZXJvV1tdID0geydpJywnVCcsJ0wnLCdaJywnZScsJ3InLCdvJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpVGltZVByZWZpeFdbXSA9IHsnaScsJ1QnLCdpJywnbScsJ2UnLCdQJywncicsJ2UnLCdmJywnaScsJ3gnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlUaW1lV1tdID0geydpJywnVCcsJ2knLCdtJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgczExNTlXW10gPSB7J3MnLCcxJywnMScsJzUnLCc5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzMjM1OVdbXSA9IHsncycsJzInLCczJywnNScsJzknLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNDb3VudHJ5V1tdID0geydzJywnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0N1cnJlbmN5V1tdID0geydzJywnQycsJ3UnLCdyJywncicsJ2UnLCduJywnYycsJ3knLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNEYXRlV1tdID0geydzJywnRCcsJ2EnLCd0JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0RlY2ltYWxXW10gPSB7J3MnLCdEJywnZScsJ2MnLCdpJywnbScsJ2EnLCdsJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzR3JvdXBpbmdXW10gPSB7J3MnLCdHJywncicsJ28nLCd1JywncCcsJ2knLCduJywnZycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0xhbmd1YWdlV1tdID0geydzJywnTCcsJ2EnLCduJywnZycsJ3UnLCdhJywnZycsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNMaXN0V1tdID0geydzJywnTCcsJ2knLCdzJywndCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0xvbmdEYXRlV1tdID0geydzJywnTCcsJ28nLCduJywnZycsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNNb25EZWNpbWFsU2VwV1tdID0geydzJywnTScsJ28nLCduJywnRCcsJ2UnLCdjJywnaScsJ20nLCdhJywnbCcsJ1MnLCdlJywncCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc01vbkdyb3VwaW5nV1tdID0geydzJywnTScsJ28nLCduJywnRycsJ3InLCdvJywndScsJ3AnLCdpJywnbicsJ2cnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNNb25UaG91c2FuZFNlcFdbXSA9IHsncycsJ00nLCdvJywnbicsJ1QnLCdoJywnbycsJ3UnLCdzJywnYScsJ24nLCdkJywnUycsJ2UnLCdwJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTmF0aXZlRGlnaXRzV1tdID0geydzJywnTicsJ2EnLCd0JywnaScsJ3YnLCdlJywnRCcsJ2knLCdnJywnaScsJ3QnLCdzJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTmVnYXRpdmVTaWduV1tdID0geydzJywnTicsJ2UnLCdnJywnYScsJ3QnLCdpJywndicsJ2UnLCdTJywnaScsJ2cnLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzUG9zaXRpdmVTaWduV1tdID0geydzJywnUCcsJ28nLCdzJywnaScsJ3QnLCdpJywndicsJ2UnLCdTJywnaScsJ2cnLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzU2hvcnREYXRlV1tdID0geydzJywnUycsJ2gnLCdvJywncicsJ3QnLCdEJywnYScsJ3QnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzVGhvdXNhbmRXW10gPSB7J3MnLCdUJywnaCcsJ28nLCd1JywncycsJ2EnLCduJywnZCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1RpbWVGb3JtYXRXW10gPSB7J3MnLCdUJywnaScsJ20nLCdlJywnRicsJ28nLCdyJywnbScsJ2EnLCd0JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzVGltZVdbXSA9IHsncycsJ1QnLCdpJywnbScsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNZZWFyTW9udGhXW10gPSB7J3MnLCdZJywnZScsJ2EnLCdyJywnTScsJ28nLCduJywndCcsJ2gnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIE51bVNoYXBlV1tdID0geydOJywndScsJ20nLCdzJywnaCcsJ2EnLCdwJywnZScsMH07CgogICAgc3dpdGNoIChsY3R5cGUpCiAgICB7CiAgICAvKiBUaGVzZSB2YWx1ZXMgYXJlIHVzZWQgYnkgU2V0TG9jYWxlSW5mbyBhbmQgR2V0TG9jYWxlSW5mbywgYW5kCiAgICAgKiB0aGUgdmFsdWVzIGFyZSBzdG9yZWQgaW4gdGhlIHJlZ2lzdHJ5LCBjb25maXJtZWQgdW5kZXIgV2luZG93cy4KICAgICAqLwogICAgY2FzZSBMT0NBTEVfSUNBTEVOREFSVFlQRTogICAgcmV0dXJuIGlDYWxlbmRhclR5cGVXOwogICAgY2FzZSBMT0NBTEVfSUNVUlJESUdJVFM6ICAgICAgcmV0dXJuIGlDdXJyRGlnaXRzVzsKICAgIGNhc2UgTE9DQUxFX0lDVVJSRU5DWTogICAgICAgIHJldHVybiBpQ3VycmVuY3lXOwogICAgY2FzZSBMT0NBTEVfSURJR0lUUzogICAgICAgICAgcmV0dXJuIGlEaWdpdHNXOwogICAgY2FzZSBMT0NBTEVfSUZJUlNUREFZT0ZXRUVLOiAgcmV0dXJuIGlGaXJzdERheU9mV2Vla1c7CiAgICBjYXNlIExPQ0FMRV9JRklSU1RXRUVLT0ZZRUFSOiByZXR1cm4gaUZpcnN0V2Vla09mWWVhclc7CiAgICBjYXNlIExPQ0FMRV9JTFpFUk86ICAgICAgICAgICByZXR1cm4gaUxaZXJvVzsKICAgIGNhc2UgTE9DQUxFX0lNRUFTVVJFOiAgICAgICAgIHJldHVybiBpTWVhc3VyZVc7CiAgICBjYXNlIExPQ0FMRV9JTkVHQ1VSUjogICAgICAgICByZXR1cm4gaU5lZ0N1cnJXOwogICAgY2FzZSBMT0NBTEVfSU5FR05VTUJFUjogICAgICAgcmV0dXJuIGlOZWdOdW1iZXJXOwogICAgY2FzZSBMT0NBTEVfSVBBUEVSU0laRTogICAgICAgcmV0dXJuIGlQYXBlclNpemVXOwogICAgY2FzZSBMT0NBTEVfSVRJTUU6ICAgICAgICAgICAgcmV0dXJuIGlUaW1lVzsKICAgIGNhc2UgTE9DQUxFX1MxMTU5OiAgICAgICAgICAgIHJldHVybiBzMTE1OVc7CiAgICBjYXNlIExPQ0FMRV9TMjM1OTogICAgICAgICAgICByZXR1cm4gczIzNTlXOwogICAgY2FzZSBMT0NBTEVfU0NVUlJFTkNZOiAgICAgICAgcmV0dXJuIHNDdXJyZW5jeVc7CiAgICBjYXNlIExPQ0FMRV9TREFURTogICAgICAgICAgICByZXR1cm4gc0RhdGVXOwogICAgY2FzZSBMT0NBTEVfU0RFQ0lNQUw6ICAgICAgICAgcmV0dXJuIHNEZWNpbWFsVzsKICAgIGNhc2UgTE9DQUxFX1NHUk9VUElORzogICAgICAgIHJldHVybiBzR3JvdXBpbmdXOwogICAgY2FzZSBMT0NBTEVfU0xJU1Q6ICAgICAgICAgICAgcmV0dXJuIHNMaXN0VzsKICAgIGNhc2UgTE9DQUxFX1NMT05HREFURTogICAgICAgIHJldHVybiBzTG9uZ0RhdGVXOwogICAgY2FzZSBMT0NBTEVfU01PTkRFQ0lNQUxTRVA6ICAgcmV0dXJuIHNNb25EZWNpbWFsU2VwVzsKICAgIGNhc2UgTE9DQUxFX1NNT05HUk9VUElORzogICAgIHJldHVybiBzTW9uR3JvdXBpbmdXOwogICAgY2FzZSBMT0NBTEVfU01PTlRIT1VTQU5EU0VQOiAgcmV0dXJuIHNNb25UaG91c2FuZFNlcFc7CiAgICBjYXNlIExPQ0FMRV9TTkVHQVRJVkVTSUdOOiAgICByZXR1cm4gc05lZ2F0aXZlU2lnblc7CiAgICBjYXNlIExPQ0FMRV9TUE9TSVRJVkVTSUdOOiAgICByZXR1cm4gc1Bvc2l0aXZlU2lnblc7CiAgICBjYXNlIExPQ0FMRV9TU0hPUlREQVRFOiAgICAgICByZXR1cm4gc1Nob3J0RGF0ZVc7CiAgICBjYXNlIExPQ0FMRV9TVEhPVVNBTkQ6ICAgICAgICByZXR1cm4gc1Rob3VzYW5kVzsKICAgIGNhc2UgTE9DQUxFX1NUSU1FOiAgICAgICAgICAgIHJldHVybiBzVGltZVc7CiAgICBjYXNlIExPQ0FMRV9TVElNRUZPUk1BVDogICAgICByZXR1cm4gc1RpbWVGb3JtYXRXOwogICAgY2FzZSBMT0NBTEVfU1lFQVJNT05USDogICAgICAgcmV0dXJuIHNZZWFyTW9udGhXOwoKICAgIC8qIFRoZSBmb2xsb3dpbmcgYXJlIG5vdCBsaXN0ZWQgdW5kZXIgTVNETiBhcyBzdXBwb3J0ZWQsCiAgICAgKiBidXQgc2VlbSB0byBiZSB1c2VkIGFuZCBhbHNvIHN0b3JlZCBpbiB0aGUgcmVnaXN0cnkuCiAgICAgKi8KICAgIGNhc2UgTE9DQUxFX0lDT1VOVFJZOiAgICAgICAgIHJldHVybiBpQ291bnRyeVc7CiAgICBjYXNlIExPQ0FMRV9JREFURTogICAgICAgICAgICByZXR1cm4gaURhdGVXOwogICAgY2FzZSBMT0NBTEVfSUxEQVRFOiAgICAgICAgICAgcmV0dXJuIGlMRGF0ZVc7CiAgICBjYXNlIExPQ0FMRV9JVExaRVJPOiAgICAgICAgICByZXR1cm4gaVRMWmVyb1c7CiAgICBjYXNlIExPQ0FMRV9TQ09VTlRSWTogICAgICAgICByZXR1cm4gc0NvdW50cnlXOwogICAgY2FzZSBMT0NBTEVfU0xBTkdVQUdFOiAgICAgICAgcmV0dXJuIHNMYW5ndWFnZVc7CgogICAgLyogVGhlIGZvbGxvd2luZyBhcmUgdXNlZCBpbiBYUCBhbmQgbGF0ZXIgKi8KICAgIGNhc2UgTE9DQUxFX0lESUdJVFNVQlNUSVRVVElPTjogcmV0dXJuIE51bVNoYXBlVzsKICAgIGNhc2UgTE9DQUxFX1NOQVRJVkVESUdJVFM6ICAgICAgcmV0dXJuIHNOYXRpdmVEaWdpdHNXOwogICAgY2FzZSBMT0NBTEVfSVRJTUVNQVJLUE9TTjogICAgICByZXR1cm4gaVRpbWVQcmVmaXhXOwogICAgfQogICAgcmV0dXJuIE5VTEw7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8KICoKICogUmV0cmlldmUgdXNlci1tb2RpZmllZCBsb2NhbGUgaW5mbyBmcm9tIHRoZSByZWdpc3RyeS4KICogUmV0dXJuIGxlbmd0aCwgMCBvbiBlcnJvciwgLTEgaWYgbm90IGZvdW5kLgogKi8Kc3RhdGljIElOVCBnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8oIExQQ1dTVFIgdmFsdWUsIExQV1NUUiBidWZmZXIsIElOVCBsZW4gKQp7CiAgICBEV09SRCBzaXplOwogICAgSU5UIHJldDsKICAgIEhBTkRMRSBoa2V5OwogICAgTlRTVEFUVVMgc3RhdHVzOwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVc7CiAgICBLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqaW5mbzsKICAgIHN0YXRpYyBjb25zdCBpbnQgaW5mb19zaXplID0gRklFTERfT0ZGU0VUKEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OLCBEYXRhKTsKCiAgICBpZiAoIShoa2V5ID0gY3JlYXRlX3JlZ2lzdHJ5X2tleSgpKSkgcmV0dXJuIC0xOwoKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmbmFtZVcsIHZhbHVlICk7CiAgICBzaXplID0gaW5mb19zaXplICsgbGVuICogc2l6ZW9mKFdDSEFSKTsKCiAgICBpZiAoIShpbmZvID0gSGVhcEFsbG9jKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBzaXplICkpKQogICAgewogICAgICAgIE50Q2xvc2UoIGhrZXkgKTsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgc3RhdHVzID0gTnRRdWVyeVZhbHVlS2V5KCBoa2V5LCAmbmFtZVcsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLCBpbmZvLCBzaXplLCAmc2l6ZSApOwogICAgaWYgKHN0YXR1cyA9PSBTVEFUVVNfQlVGRkVSX09WRVJGTE9XICYmICFidWZmZXIpIHN0YXR1cyA9IDA7CgogICAgaWYgKCFzdGF0dXMpCiAgICB7CiAgICAgICAgcmV0ID0gKHNpemUgLSBpbmZvX3NpemUpIC8gc2l6ZW9mKFdDSEFSKTsKICAgICAgICAvKiBhcHBlbmQgdGVybWluYXRpbmcgbnVsbCBpZiBuZWVkZWQgKi8KICAgICAgICBpZiAoIXJldCB8fCAoKFdDSEFSICopaW5mby0+RGF0YSlbcmV0LTFdKQogICAgICAgIHsKICAgICAgICAgICAgaWYgKHJldCA8IGxlbiB8fCAhYnVmZmVyKSByZXQrKzsKICAgICAgICAgICAgZWxzZQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsKICAgICAgICAgICAgICAgIHJldCA9IDA7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgaWYgKHJldCAmJiBidWZmZXIpCiAgICAgICAgewogICAgICAgICAgICBtZW1jcHkoIGJ1ZmZlciwgaW5mby0+RGF0YSwgKHJldC0xKSAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICAgICAgYnVmZmVyW3JldC0xXSA9IDA7CiAgICAgICAgfQogICAgfQogICAgZWxzZQogICAgewogICAgICAgIGlmIChzdGF0dXMgPT0gU1RBVFVTX09CSkVDVF9OQU1FX05PVF9GT1VORCkgcmV0ID0gLTE7CiAgICAgICAgZWxzZQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBSdGxOdFN0YXR1c1RvRG9zRXJyb3Ioc3RhdHVzKSApOwogICAgICAgICAgICByZXQgPSAwOwogICAgICAgIH0KICAgIH0KICAgIE50Q2xvc2UoIGhrZXkgKTsKICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBpbmZvICk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0TG9jYWxlSW5mb0EgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBpbmZvcm1hdGlvbiBhYm91dCBhbiBhc3BlY3Qgb2YgYSBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUKICogIGxjdHlwZSBbSV0gTENUWVBFXyBmbGFncyBmcm9tICJ3aW5ubHMuaCIKICogIGJ1ZmZlciBbT10gRGVzdGluYXRpb24gZm9yIHRoZSBpbmZvcm1hdGlvbgogKiAgbGVuICAgIFtJXSBMZW5ndGggb2YgYnVmZmVyIGluIGNoYXJhY3RlcnMKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVGhlIHNpemUgb2YgdGhlIGRhdGEgcmVxdWVzdGVkLiBJZiBidWZmZXIgaXMgbm9uLU5VTEwsIGl0IGlzIGZpbGxlZAogKiAgICAgICAgICAgd2l0aCB0aGUgaW5mb3JtYXRpb24uCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICogTk9URVMKICogIC0gTE9DQUxFX05FVVRSQUwgaXMgZXF1YWwgdG8gTE9DQUxFX1NZU1RFTV9ERUZBVUxUCiAqICAtIFRoZSBzdHJpbmcgcmV0dXJuZWQgaXMgTlVMIHRlcm1pbmF0ZWQsIGV4Y2VwdCBmb3IgTE9DQUxFX0ZPTlRTSUdOQVRVUkUsCiAqICAgIHdoaWNoIGlzIGEgYml0IHN0cmluZy4KICovCklOVCBXSU5BUEkgR2V0TG9jYWxlSW5mb0EoIExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBTVFIgYnVmZmVyLCBJTlQgbGVuICkKewogICAgV0NIQVIgKmJ1ZmZlclc7CiAgICBJTlQgbGVuVywgcmV0OwoKICAgIFRSQUNFKCAiKGxjaWQ9MHglbHgsbGN0eXBlPTB4JWx4LCVwLCVkKVxuIiwgbGNpZCwgbGN0eXBlLCBidWZmZXIsIGxlbiApOwoKICAgIGlmIChsZW4gPCAwIHx8IChsZW4gJiYgIWJ1ZmZlcikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgaWYgKCFsZW4pIGJ1ZmZlciA9IE5VTEw7CgogICAgaWYgKCEobGVuVyA9IEdldExvY2FsZUluZm9XKCBsY2lkLCBsY3R5cGUsIE5VTEwsIDAgKSkpIHJldHVybiAwOwoKICAgIGlmICghKGJ1ZmZlclcgPSBIZWFwQWxsb2MoIEdldFByb2Nlc3NIZWFwKCksIDAsIGxlblcgKiBzaXplb2YoV0NIQVIpICkpKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkgKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIGlmICgocmV0ID0gR2V0TG9jYWxlSW5mb1coIGxjaWQsIGxjdHlwZSwgYnVmZmVyVywgbGVuVyApKSkKICAgIHsKICAgICAgICBpZiAoKGxjdHlwZSAmIExPQ0FMRV9SRVRVUk5fTlVNQkVSKSB8fAogICAgICAgICAgICAoKGxjdHlwZSAmIH5MT0NBTEVfTE9DQUxFSU5GT0ZMQUdTTUFTSykgPT0gTE9DQUxFX0ZPTlRTSUdOQVRVUkUpKQogICAgICAgIHsKICAgICAgICAgICAgLyogaXQncyBub3QgYW4gQVNDSUkgc3RyaW5nLCBqdXN0IGJ5dGVzICovCiAgICAgICAgICAgIHJldCAqPSBzaXplb2YoV0NIQVIpOwogICAgICAgICAgICBpZiAoYnVmZmVyKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBpZiAocmV0IDw9IGxlbikgbWVtY3B5KCBidWZmZXIsIGJ1ZmZlclcsIHJldCApOwogICAgICAgICAgICAgICAgZWxzZQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUiApOwogICAgICAgICAgICAgICAgICAgIHJldCA9IDA7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgIHsKICAgICAgICAgICAgVUlOVCBjb2RlcGFnZSA9IENQX0FDUDsKICAgICAgICAgICAgaWYgKCEobGN0eXBlICYgTE9DQUxFX1VTRV9DUF9BQ1ApKSBjb2RlcGFnZSA9IGdldF9sY2lkX2NvZGVwYWdlKCBsY2lkICk7CiAgICAgICAgICAgIHJldCA9IFdpZGVDaGFyVG9NdWx0aUJ5dGUoIGNvZGVwYWdlLCAwLCBidWZmZXJXLCByZXQsIGJ1ZmZlciwgbGVuLCBOVUxMLCBOVUxMICk7CiAgICAgICAgfQogICAgfQogICAgSGVhcEZyZWUoIEdldFByb2Nlc3NIZWFwKCksIDAsIGJ1ZmZlclcgKTsKICAgIHJldHVybiByZXQ7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRMb2NhbGVJbmZvVyAoS0VSTkVMMzIuQCkKICoKICogU2VlIEdldExvY2FsZUluZm9BLgogKi8KSU5UIFdJTkFQSSBHZXRMb2NhbGVJbmZvVyggTENJRCBsY2lkLCBMQ1RZUEUgbGN0eXBlLCBMUFdTVFIgYnVmZmVyLCBJTlQgbGVuICkKewogICAgTEFOR0lEIGxhbmdfaWQ7CiAgICBIUlNSQyBocnNyYzsKICAgIEhHTE9CQUwgaG1lbTsKICAgIElOVCByZXQ7CiAgICBVSU5UIGxjZmxhZ3M7CiAgICBjb25zdCBXQ0hBUiAqcDsKICAgIHVuc2lnbmVkIGludCBpOwoKICAgIGlmIChsZW4gPCAwIHx8IChsZW4gJiYgIWJ1ZmZlcikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgaWYgKCFsZW4pIGJ1ZmZlciA9IE5VTEw7CgogICAgbGNpZCA9IENvbnZlcnREZWZhdWx0TG9jYWxlKGxjaWQpOwoKICAgIGxjZmxhZ3MgPSBsY3R5cGUgJiBMT0NBTEVfTE9DQUxFSU5GT0ZMQUdTTUFTSzsKICAgIGxjdHlwZSAmPSAweGZmZmY7CgogICAgVFJBQ0UoICIobGNpZD0weCVseCxsY3R5cGU9MHglbHgsJXAsJWQpXG4iLCBsY2lkLCBsY3R5cGUsIGJ1ZmZlciwgbGVuICk7CgogICAgLyogZmlyc3QgY2hlY2sgZm9yIG92ZXJyaWRlcyBpbiB0aGUgcmVnaXN0cnkgKi8KCiAgICBpZiAoIShsY2ZsYWdzICYgTE9DQUxFX05PVVNFUk9WRVJSSURFKSAmJiBsY2lkID09IEdldFVzZXJEZWZhdWx0TENJRCgpKQogICAgewogICAgICAgIGNvbnN0IFdDSEFSICp2YWx1ZSA9IGdldF9sb2NhbGVfdmFsdWVfbmFtZShsY3R5cGUpOwoKICAgICAgICBpZiAodmFsdWUpCiAgICAgICAgewogICAgICAgICAgICBpZiAobGNmbGFncyAmIExPQ0FMRV9SRVRVUk5fTlVNQkVSKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBXQ0hBUiB0bXBbMTZdOwogICAgICAgICAgICAgICAgcmV0ID0gZ2V0X3JlZ2lzdHJ5X2xvY2FsZV9pbmZvKCB2YWx1ZSwgdG1wLCBzaXplb2YodG1wKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgICAgICAgICBpZiAocmV0ID4gMCkKICAgICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgICAgICBXQ0hBUiAqZW5kOwogICAgICAgICAgICAgICAgICAgIFVJTlQgbnVtYmVyID0gc3RydG9sVyggdG1wLCAmZW5kLCAxMCApOwogICAgICAgICAgICAgICAgICAgIGlmICgqZW5kKSAgLyogaW52YWxpZCBudW1iZXIgKi8KICAgICAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9GTEFHUyApOwogICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgcmV0ID0gc2l6ZW9mKFVJTlQpL3NpemVvZihXQ0hBUik7CiAgICAgICAgICAgICAgICAgICAgaWYgKCFidWZmZXIpIHJldHVybiByZXQ7CiAgICAgICAgICAgICAgICAgICAgaWYgKHJldCA+IGxlbikKICAgICAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUiApOwogICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgbWVtY3B5KCBidWZmZXIsICZudW1iZXIsIHNpemVvZihudW1iZXIpICk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZSByZXQgPSBnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8oIHZhbHVlLCBidWZmZXIsIGxlbiApOwoKICAgICAgICAgICAgaWYgKHJldCAhPSAtMSkgcmV0dXJuIHJldDsKICAgICAgICB9CiAgICB9CgogICAgLyogbm93IGxvYWQgaXQgZnJvbSBrZXJuZWwgcmVzb3VyY2VzICovCgogICAgbGFuZ19pZCA9IExBTkdJREZST01MQ0lEKCBsY2lkICk7CgogICAgLyogcmVwbGFjZSBTVUJMQU5HX05FVVRSQUwgYnkgU1VCTEFOR19ERUZBVUxUICovCiAgICBpZiAoU1VCTEFOR0lEKGxhbmdfaWQpID09IFNVQkxBTkdfTkVVVFJBTCkKICAgICAgICBsYW5nX2lkID0gTUFLRUxBTkdJRChQUklNQVJZTEFOR0lEKGxhbmdfaWQpLCBTVUJMQU5HX0RFRkFVTFQpOwoKICAgIGlmICghKGhyc3JjID0gRmluZFJlc291cmNlRXhXKCBrZXJuZWwzMl9oYW5kbGUsIChMUFdTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENXU1RSKSgobGN0eXBlID4+IDQpICsgMSksIGxhbmdfaWQgKSkpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX0ZMQUdTICk7ICAvKiBubyBzdWNoIGxjdHlwZSAqLwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgaWYgKCEoaG1lbSA9IExvYWRSZXNvdXJjZSgga2VybmVsMzJfaGFuZGxlLCBocnNyYyApKSkKICAgICAgICByZXR1cm4gMDsKCiAgICBwID0gTG9ja1Jlc291cmNlKCBobWVtICk7CiAgICBmb3IgKGkgPSAwOyBpIDwgKGxjdHlwZSAmIDB4MGYpOyBpKyspIHAgKz0gKnAgKyAxOwoKICAgIGlmIChsY2ZsYWdzICYgTE9DQUxFX1JFVFVSTl9OVU1CRVIpIHJldCA9IHNpemVvZihVSU5UKS9zaXplb2YoV0NIQVIpOwogICAgZWxzZSByZXQgPSAobGN0eXBlID09IExPQ0FMRV9GT05UU0lHTkFUVVJFKSA/ICpwIDogKnAgKyAxOwoKICAgIGlmICghYnVmZmVyKSByZXR1cm4gcmV0OwoKICAgIGlmIChyZXQgPiBsZW4pCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKGxjZmxhZ3MgJiBMT0NBTEVfUkVUVVJOX05VTUJFUikKICAgIHsKICAgICAgICBVSU5UIG51bWJlcjsKICAgICAgICBXQ0hBUiAqZW5kLCAqdG1wID0gSGVhcEFsbG9jKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCAoKnAgKyAxKSAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICBpZiAoIXRtcCkgcmV0dXJuIDA7CiAgICAgICAgbWVtY3B5KCB0bXAsIHAgKyAxLCAqcCAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICB0bXBbKnBdID0gMDsKICAgICAgICBudW1iZXIgPSBzdHJ0b2xXKCB0bXAsICZlbmQsIDEwICk7CiAgICAgICAgaWYgKCEqZW5kKQogICAgICAgICAgICBtZW1jcHkoIGJ1ZmZlciwgJm51bWJlciwgc2l6ZW9mKG51bWJlcikgKTsKICAgICAgICBlbHNlICAvKiBpbnZhbGlkIG51bWJlciAqLwogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX0ZMQUdTICk7CiAgICAgICAgICAgIHJldCA9IDA7CiAgICAgICAgfQogICAgICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCB0bXAgKTsKCiAgICAgICAgVFJBQ0UoICIobGNpZD0weCVseCxsY3R5cGU9MHglbHgsJXAsJWQpIHJldHVybmluZyBudW1iZXIgJWRcbiIsCiAgICAgICAgICAgICAgIGxjaWQsIGxjdHlwZSwgYnVmZmVyLCBsZW4sIG51bWJlciApOwogICAgfQogICAgZWxzZQogICAgewogICAgICAgIG1lbWNweSggYnVmZmVyLCBwICsgMSwgKnAgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgaWYgKGxjdHlwZSAhPSBMT0NBTEVfRk9OVFNJR05BVFVSRSkgYnVmZmVyW3JldC0xXSA9IDA7CgogICAgICAgIFRSQUNFKCAiKGxjaWQ9MHglbHgsbGN0eXBlPTB4JWx4LCVwLCVkKSByZXR1cm5pbmcgJWQgJXNcbiIsCiAgICAgICAgICAgICAgIGxjaWQsIGxjdHlwZSwgYnVmZmVyLCBsZW4sIHJldCwgZGVidWdzdHJfdyhidWZmZXIpICk7CiAgICB9CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJU2V0TG9jYWxlSW5mb0EJW0tFUk5FTDMyLkBdCiAqCiAqIFNldCBpbmZvcm1hdGlvbiBhYm91dCBhbiBhc3BlY3Qgb2YgYSBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUKICogIGxjdHlwZSBbSV0gTENUWVBFXyBmbGFncyBmcm9tICJ3aW5ubHMuaCIKICogIGRhdGEgICBbSV0gSW5mb3JtYXRpb24gdG8gc2V0CiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIFRoZSBpbmZvcm1hdGlvbiBnaXZlbiB3aWxsIGJlIHJldHVybmVkIGJ5IEdldExvY2FsZUluZm9BKCkKICogICAgICAgICAgIHdoZW5ldmVyIGl0IGlzIGNhbGxlZCB3aXRob3V0IExPQ0FMRV9OT1VTRVJPVkVSUklERS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICogTk9URVMKICogIC0gVmFsdWVzIGFyZSBvbmx5IGJlIHNldCBmb3IgdGhlIGN1cnJlbnQgdXNlciBsb2NhbGU7IHRoZSBzeXN0ZW0gbG9jYWxlCiAqICBzZXR0aW5ncyBjYW5ub3QgYmUgY2hhbmdlZC4KICogIC0gQW55IHNldHRpbmdzIGNoYW5nZWQgYnkgdGhpcyBjYWxsIGFyZSBsb3N0IHdoZW4gdGhlIGxvY2FsZSBpcyBjaGFuZ2VkIGJ5CiAqICB0aGUgY29udHJvbCBwYW5lbCAoaW4gV2luZSwgdGhpcyBoYXBwZW5zIGV2ZXJ5IHRpbWUgeW91IGNoYW5nZSBMQU5HKS4KICogIC0gVGhlIG5hdGl2ZSBpbXBsZW1lbnRhdGlvbiBvZiB0aGlzIGZ1bmN0aW9uIGRvZXMgbm90IGNoZWNrIHRoYXQgbGNpZCBtYXRjaGVzCiAqICB0aGUgY3VycmVudCB1c2VyIGxvY2FsZSwgYW5kIHNpbXBseSBzZXRzIHRoZSBuZXcgdmFsdWVzLiBXaW5lIHdhcm5zIHlvdSBpbgogKiAgdGhpcyBjYXNlLCBidXQgYmVoYXZlcyB0aGUgc2FtZS4KICovCkJPT0wgV0lOQVBJIFNldExvY2FsZUluZm9BKExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBDU1RSIGRhdGEpCnsKICAgIFVJTlQgY29kZXBhZ2UgPSBDUF9BQ1A7CiAgICBXQ0hBUiAqc3RyVzsKICAgIERXT1JEIGxlbjsKICAgIEJPT0wgcmV0OwoKICAgIGxjaWQgPSBDb252ZXJ0RGVmYXVsdExvY2FsZShsY2lkKTsKCiAgICBpZiAoIShsY3R5cGUgJiBMT0NBTEVfVVNFX0NQX0FDUCkpIGNvZGVwYWdlID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKCiAgICBpZiAoIWRhdGEpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KICAgIGxlbiA9IE11bHRpQnl0ZVRvV2lkZUNoYXIoIGNvZGVwYWdlLCAwLCBkYXRhLCAtMSwgTlVMTCwgMCApOwogICAgaWYgKCEoc3RyVyA9IEhlYXBBbGxvYyggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGVuICogc2l6ZW9mKFdDSEFSKSApKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgTXVsdGlCeXRlVG9XaWRlQ2hhciggY29kZXBhZ2UsIDAsIGRhdGEsIC0xLCBzdHJXLCBsZW4gKTsKICAgIHJldCA9IFNldExvY2FsZUluZm9XKCBsY2lkLCBsY3R5cGUsIHN0clcgKTsKICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHJXICk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJU2V0TG9jYWxlSW5mb1cJKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBTZXRMb2NhbGVJbmZvQS4KICovCkJPT0wgV0lOQVBJIFNldExvY2FsZUluZm9XKCBMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUsIExQQ1dTVFIgZGF0YSApCnsKICAgIGNvbnN0IFdDSEFSICp2YWx1ZTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpbnRsV1tdID0geydpJywnbicsJ3QnLCdsJywwIH07CiAgICBVTklDT0RFX1NUUklORyB2YWx1ZVc7CiAgICBOVFNUQVRVUyBzdGF0dXM7CiAgICBIQU5ETEUgaGtleTsKCiAgICBsY2lkID0gQ29udmVydERlZmF1bHRMb2NhbGUobGNpZCk7CgogICAgbGN0eXBlICY9IDB4ZmZmZjsKICAgIHZhbHVlID0gZ2V0X2xvY2FsZV92YWx1ZV9uYW1lKCBsY3R5cGUgKTsKCiAgICBpZiAoIWRhdGEgfHwgIXZhbHVlKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaWYgKGxjdHlwZSA9PSBMT0NBTEVfSURBVEUgfHwgbGN0eXBlID09IExPQ0FMRV9JTERBVEUpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX0ZMQUdTICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmIChsY2lkICE9IEdldFVzZXJEZWZhdWx0TENJRCgpKQogICAgewogICAgICAgIC8qIFdpbmRvd3MgZG9lcyBub3QgY2hlY2sgdGhhdCB0aGUgbGNpZCBtYXRjaGVzIHRoZSBjdXJyZW50IGxjaWQgKi8KICAgICAgICBXQVJOKCJsb2NhbGUgMHglMDhseCBpc24ndCB0aGUgY3VycmVudCBsb2NhbGUgKDB4JTA4bHgpLCBzZXR0aW5nIGFueXdheSFcbiIsCiAgICAgICAgICAgICBsY2lkLCBHZXRVc2VyRGVmYXVsdExDSUQoKSk7CiAgICB9CgogICAgVFJBQ0UoInNldHRpbmcgJWx4ICglcykgdG8gJXNcbiIsIGxjdHlwZSwgZGVidWdzdHJfdyh2YWx1ZSksIGRlYnVnc3RyX3coZGF0YSkgKTsKCiAgICAvKiBGSVhNRTogc2hvdWxkIGNoZWNrIHRoYXQgZGF0YSB0byBzZXQgaXMgc2FuZSAqLwoKICAgIC8qIEZJWE1FOiBwcm9maWxlIGZ1bmN0aW9ucyBzaG91bGQgbWFwIHRvIHJlZ2lzdHJ5ICovCiAgICBXcml0ZVByb2ZpbGVTdHJpbmdXKCBpbnRsVywgdmFsdWUsIGRhdGEgKTsKCiAgICBpZiAoIShoa2V5ID0gY3JlYXRlX3JlZ2lzdHJ5X2tleSgpKSkgcmV0dXJuIEZBTFNFOwogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZ2YWx1ZVcsIHZhbHVlICk7CiAgICBzdGF0dXMgPSBOdFNldFZhbHVlS2V5KCBoa2V5LCAmdmFsdWVXLCAwLCBSRUdfU1osIGRhdGEsIChzdHJsZW5XKGRhdGEpKzEpKnNpemVvZihXQ0hBUikgKTsKCiAgICBpZiAobGN0eXBlID09IExPQ0FMRV9TU0hPUlREQVRFIHx8IGxjdHlwZSA9PSBMT0NBTEVfU0xPTkdEQVRFKQogICAgewogICAgICAvKiBTZXQgSS12YWx1ZSBmcm9tIFMgdmFsdWUgKi8KICAgICAgV0NIQVIgKmxwRCwgKmxwTSwgKmxwWTsKICAgICAgV0NIQVIgc3pCdWZmWzJdOwoKICAgICAgbHBEID0gc3RycmNoclcoZGF0YSwgJ2QnKTsKICAgICAgbHBNID0gc3RycmNoclcoZGF0YSwgJ00nKTsKICAgICAgbHBZID0gc3RycmNoclcoZGF0YSwgJ3knKTsKCiAgICAgIGlmIChscEQgPD0gbHBNKQogICAgICB7CiAgICAgICAgc3pCdWZmWzBdID0gJzEnOyAvKiBELU0tWSAqLwogICAgICB9CiAgICAgIGVsc2UKICAgICAgewogICAgICAgIGlmIChscFkgPD0gbHBNKQogICAgICAgICAgc3pCdWZmWzBdID0gJzInOyAvKiBZLU0tRCAqLwogICAgICAgIGVsc2UKICAgICAgICAgIHN6QnVmZlswXSA9ICcwJzsgLyogTS1ELVkgKi8KICAgICAgfQoKICAgICAgc3pCdWZmWzFdID0gJ1wwJzsKCiAgICAgIGlmIChsY3R5cGUgPT0gTE9DQUxFX1NTSE9SVERBVEUpCiAgICAgICAgbGN0eXBlID0gTE9DQUxFX0lEQVRFOwogICAgICBlbHNlCiAgICAgICAgbGN0eXBlID0gTE9DQUxFX0lMREFURTsKCiAgICAgIHZhbHVlID0gZ2V0X2xvY2FsZV92YWx1ZV9uYW1lKCBsY3R5cGUgKTsKCiAgICAgIFdyaXRlUHJvZmlsZVN0cmluZ1coIGludGxXLCB2YWx1ZSwgc3pCdWZmICk7CgogICAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJnZhbHVlVywgdmFsdWUgKTsKICAgICAgc3RhdHVzID0gTnRTZXRWYWx1ZUtleSggaGtleSwgJnZhbHVlVywgMCwgUkVHX1NaLCBzekJ1ZmYsIHNpemVvZihzekJ1ZmYpICk7CiAgICB9CgogICAgTnRDbG9zZSggaGtleSApOwoKICAgIGlmIChzdGF0dXMpIFNldExhc3RFcnJvciggUnRsTnRTdGF0dXNUb0Rvc0Vycm9yKHN0YXR1cykgKTsKICAgIHJldHVybiAhc3RhdHVzOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgICAgR2V0QUNQICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgY3VycmVudCBBbnNpIGNvZGUgcGFnZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICAgIFRoZSBjdXJyZW50IEFuc2kgY29kZSBwYWdlIGlkZW50aWZpZXIgZm9yIHRoZSBzeXN0ZW0uCiAqLwpVSU5UIFdJTkFQSSBHZXRBQ1Aodm9pZCkKewogICAgYXNzZXJ0KCBhbnNpX2NwdGFibGUgKTsKICAgIHJldHVybiBhbnNpX2NwdGFibGUtPmluZm8uY29kZXBhZ2U7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBTZXRDUEdsb2JhbCAgIChLRVJORUwzMi5AKQogKgogKiBTZXQgdGhlIGN1cnJlbnQgQW5zaSBjb2RlIHBhZ2UgSWQgZm9yIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgICBhY3AgW0ldIGNvZGUgcGFnZSBJRCB0byBiZSB0aGUgbmV3IEFDUC4KICoKICogUkVUVVJOUwogKiAgICBUaGUgcHJldmlvdXMgQUNQLgogKi8KVUlOVCBXSU5BUEkgU2V0Q1BHbG9iYWwoIFVJTlQgYWNwICkKewogICAgVUlOVCByZXQgPSBHZXRBQ1AoKTsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKm5ld19jcHRhYmxlID0gd2luZV9jcF9nZXRfdGFibGUoIGFjcCApOwoKICAgIGlmIChuZXdfY3B0YWJsZSkgYW5zaV9jcHRhYmxlID0gbmV3X2NwdGFibGU7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBHZXRPRU1DUCAgIChLRVJORUwzMi5AKQogKgogKiBHZXQgdGhlIGN1cnJlbnQgT0VNIGNvZGUgcGFnZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICAgIFRoZSBjdXJyZW50IE9FTSBjb2RlIHBhZ2UgaWRlbnRpZmllciBmb3IgdGhlIHN5c3RlbS4KICovClVJTlQgV0lOQVBJIEdldE9FTUNQKHZvaWQpCnsKICAgIGFzc2VydCggb2VtX2NwdGFibGUgKTsKICAgIHJldHVybiBvZW1fY3B0YWJsZS0+aW5mby5jb2RlcGFnZTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSXNWYWxpZENvZGVQYWdlICAgKEtFUk5FTDMyLkApCiAqCiAqIERldGVybWluZSBpZiBhIGdpdmVuIGNvZGUgcGFnZSBpZGVudGlmaWVyIGlzIHZhbGlkLgogKgogKiBQQVJBTVMKICogIGNvZGVwYWdlIFtJXSBDb2RlIHBhZ2UgSWQgdG8gdmVyaWZ5LgogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBJZiBjb2RlcGFnZSBpcyB2YWxpZCBhbmQgYXZhaWxhYmxlIG9uIHRoZSBzeXN0ZW0sCiAqICBGQUxTRSBvdGhlcndpc2UuCiAqLwpCT09MIFdJTkFQSSBJc1ZhbGlkQ29kZVBhZ2UoIFVJTlQgY29kZXBhZ2UgKQp7CiAgICBzd2l0Y2goY29kZXBhZ2UpIHsKICAgIGNhc2UgQ1BfVVRGNzoKICAgIGNhc2UgQ1BfVVRGODoKICAgICAgICByZXR1cm4gVFJVRTsKICAgIGRlZmF1bHQ6CiAgICAgICAgcmV0dXJuIHdpbmVfY3BfZ2V0X3RhYmxlKCBjb2RlcGFnZSApICE9IE5VTEw7CiAgICB9Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIElzREJDU0xlYWRCeXRlRXggICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgY2hhcmFjdGVyIGlzIGEgbGVhZCBieXRlIGluIGEgZ2l2ZW4gY29kZSBwYWdlLgogKgogKiBQQVJBTVMKICogIGNvZGVwYWdlIFtJXSBDb2RlIHBhZ2UgZm9yIHRoZSB0ZXN0LgogKiAgdGVzdGNoYXIgW0ldIENoYXJhY3RlciB0byB0ZXN0CiAqCiAqIFJFVFVSTlMKICogIFRSVUUsIGlmIHRlc3RjaGFyIGlzIGEgbGVhZCBieXRlIGluIGNvZGVwYWdlLAogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNEQkNTTGVhZEJ5dGVFeCggVUlOVCBjb2RlcGFnZSwgQllURSB0ZXN0Y2hhciApCnsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBjb2RlcGFnZSApOwogICAgcmV0dXJuIHRhYmxlICYmIGlzX2RiY3NfbGVhZGJ5dGUoIHRhYmxlLCB0ZXN0Y2hhciApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc0RCQ1NMZWFkQnl0ZSAgIChLRVJORUwzMi5AKQogKiAgICAgICAgICAgSXNEQkNTTGVhZEJ5dGUgICAoS0VSTkVMLjIwNykKICoKICogRGV0ZXJtaW5lIGlmIGEgY2hhcmFjdGVyIGlzIGEgbGVhZCBieXRlLgogKgogKiBQQVJBTVMKICogIHRlc3RjaGFyIFtJXSBDaGFyYWN0ZXIgdG8gdGVzdAogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBpZiB0ZXN0Y2hhciBpcyBhIGxlYWQgYnl0ZSBpbiB0aGUgQW5zaWkgY29kZSBwYWdlLAogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNEQkNTTGVhZEJ5dGUoIEJZVEUgdGVzdGNoYXIgKQp7CiAgICBpZiAoIWFuc2lfY3B0YWJsZSkgcmV0dXJuIEZBTFNFOwogICAgcmV0dXJuIGlzX2RiY3NfbGVhZGJ5dGUoIGFuc2lfY3B0YWJsZSwgdGVzdGNoYXIgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0Q1BJbmZvICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBpbmZvcm1hdGlvbiBhYm91dCBhIGNvZGUgcGFnZS4KICoKICogUEFSQU1TCiAqICBjb2RlcGFnZSBbSV0gQ29kZSBwYWdlIG51bWJlcgogKiAgY3BpbmZvICAgW09dIERlc3RpbmF0aW9uIGZvciBjb2RlIHBhZ2UgaW5mb3JtYXRpb24KICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4gY3BpbmZvIGlzIHVwZGF0ZWQgd2l0aCB0aGUgaW5mb3JtYXRpb24gYWJvdXQgY29kZXBhZ2UuCiAqICBGYWlsdXJlOiBGQUxTRSwgaWYgY29kZXBhZ2UgaXMgaW52YWxpZCBvciBjcGluZm8gaXMgTlVMTC4KICovCkJPT0wgV0lOQVBJIEdldENQSW5mbyggVUlOVCBjb2RlcGFnZSwgTFBDUElORk8gY3BpbmZvICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CgogICAgaWYgKCFjcGluZm8pCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAoISh0YWJsZSA9IGdldF9jb2RlcGFnZV90YWJsZSggY29kZXBhZ2UgKSkpCiAgICB7CiAgICAgICAgc3dpdGNoKGNvZGVwYWdlKQogICAgICAgIHsKICAgICAgICAgICAgY2FzZSBDUF9VVEY3OgogICAgICAgICAgICBjYXNlIENQX1VURjg6CiAgICAgICAgICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzBdID0gMHgzZjsKICAgICAgICAgICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMV0gPSAwOwogICAgICAgICAgICAgICAgY3BpbmZvLT5MZWFkQnl0ZVswXSA9IGNwaW5mby0+TGVhZEJ5dGVbMV0gPSAwOwogICAgICAgICAgICAgICAgY3BpbmZvLT5NYXhDaGFyU2l6ZSA9IChjb2RlcGFnZSA9PSBDUF9VVEY3KSA/IDUgOiA0OwogICAgICAgICAgICAgICAgcmV0dXJuIFRSVUU7CiAgICAgICAgfQoKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgaWYgKHRhYmxlLT5pbmZvLmRlZl9jaGFyICYgMHhmZjAwKQogICAgewogICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMF0gPSB0YWJsZS0+aW5mby5kZWZfY2hhciAmIDB4ZmYwMDsKICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzFdID0gdGFibGUtPmluZm8uZGVmX2NoYXIgJiAweDAwZmY7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclswXSA9IHRhYmxlLT5pbmZvLmRlZl9jaGFyICYgMHhmZjsKICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzFdID0gMDsKICAgIH0KICAgIGlmICgoY3BpbmZvLT5NYXhDaGFyU2l6ZSA9IHRhYmxlLT5pbmZvLmNoYXJfc2l6ZSkgPT0gMikKICAgICAgICBtZW1jcHkoIGNwaW5mby0+TGVhZEJ5dGUsIHRhYmxlLT5kYmNzLmxlYWRfYnl0ZXMsIHNpemVvZihjcGluZm8tPkxlYWRCeXRlKSApOwogICAgZWxzZQogICAgICAgIGNwaW5mby0+TGVhZEJ5dGVbMF0gPSBjcGluZm8tPkxlYWRCeXRlWzFdID0gMDsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRDUEluZm9FeEEgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGV4dGVuZGVkIGluZm9ybWF0aW9uIGFib3V0IGEgY29kZSBwYWdlLgogKgogKiBQQVJBTVMKICogIGNvZGVwYWdlIFtJXSBDb2RlIHBhZ2UgbnVtYmVyCiAqICBkd0ZsYWdzICBbSV0gUmVzZXJ2ZWQsIG11c3QgdG8gMC4KICogIGNwaW5mbyAgIFtPXSBEZXN0aW5hdGlvbiBmb3IgY29kZSBwYWdlIGluZm9ybWF0aW9uCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIGNwaW5mbyBpcyB1cGRhdGVkIHdpdGggdGhlIGluZm9ybWF0aW9uIGFib3V0IGNvZGVwYWdlLgogKiAgRmFpbHVyZTogRkFMU0UsIGlmIGNvZGVwYWdlIGlzIGludmFsaWQgb3IgY3BpbmZvIGlzIE5VTEwuCiAqLwpCT09MIFdJTkFQSSBHZXRDUEluZm9FeEEoIFVJTlQgY29kZXBhZ2UsIERXT1JEIGR3RmxhZ3MsIExQQ1BJTkZPRVhBIGNwaW5mbyApCnsKICAgIENQSU5GT0VYVyBjcGluZm9XOwoKICAgIGlmICghR2V0Q1BJbmZvRXhXKCBjb2RlcGFnZSwgZHdGbGFncywgJmNwaW5mb1cgKSkKICAgICAgcmV0dXJuIEZBTFNFOwoKICAgIC8qIHRoZSBsYXlvdXQgaXMgdGhlIHNhbWUgZXhjZXB0IGZvciBDb2RlUGFnZU5hbWUgKi8KICAgIG1lbWNweShjcGluZm8sICZjcGluZm9XLCBzaXplb2YoQ1BJTkZPRVhBKSk7CiAgICBXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgY3BpbmZvVy5Db2RlUGFnZU5hbWUsIC0xLCBjcGluZm8tPkNvZGVQYWdlTmFtZSwgc2l6ZW9mKGNwaW5mby0+Q29kZVBhZ2VOYW1lKSwgTlVMTCwgTlVMTCk7CiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRDUEluZm9FeFcgICAoS0VSTkVMMzIuQCkKICoKICogVW5pY29kZSB2ZXJzaW9uIG9mIEdldENQSW5mb0V4QS4KICovCkJPT0wgV0lOQVBJIEdldENQSW5mb0V4VyggVUlOVCBjb2RlcGFnZSwgRFdPUkQgZHdGbGFncywgTFBDUElORk9FWFcgY3BpbmZvICkKewogICAgaWYgKCFHZXRDUEluZm8oIGNvZGVwYWdlLCAoTFBDUElORk8pY3BpbmZvICkpCiAgICAgIHJldHVybiBGQUxTRTsKCiAgICBzd2l0Y2goY29kZXBhZ2UpCiAgICB7CiAgICAgICAgY2FzZSBDUF9VVEY3OgogICAgICAgIHsKICAgICAgICAgICAgc3RhdGljIGNvbnN0IFdDSEFSIHV0ZjdbXSA9IHsnVScsJ24nLCdpJywnYycsJ28nLCdkJywnZScsJyAnLCcoJywnVScsJ1QnLCdGJywnLScsJzcnLCcpJywwfTsKCiAgICAgICAgICAgIGNwaW5mby0+Q29kZVBhZ2UgPSBDUF9VVEY3OwogICAgICAgICAgICBjcGluZm8tPlVuaWNvZGVEZWZhdWx0Q2hhciA9IDB4M2Y7CiAgICAgICAgICAgIHN0cmNweVcoY3BpbmZvLT5Db2RlUGFnZU5hbWUsIHV0ZjcpOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CgogICAgICAgIGNhc2UgQ1BfVVRGODoKICAgICAgICB7CiAgICAgICAgICAgIHN0YXRpYyBjb25zdCBXQ0hBUiB1dGY4W10gPSB7J1UnLCduJywnaScsJ2MnLCdvJywnZCcsJ2UnLCcgJywnKCcsJ1UnLCdUJywnRicsJy0nLCc4JywnKScsMH07CgogICAgICAgICAgICBjcGluZm8tPkNvZGVQYWdlID0gQ1BfVVRGODsKICAgICAgICAgICAgY3BpbmZvLT5Vbmljb2RlRGVmYXVsdENoYXIgPSAweDNmOwogICAgICAgICAgICBzdHJjcHlXKGNwaW5mby0+Q29kZVBhZ2VOYW1lLCB1dGY4KTsKICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQoKICAgICAgICBkZWZhdWx0OgogICAgICAgIHsKICAgICAgICAgICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGUgPSBnZXRfY29kZXBhZ2VfdGFibGUoIGNvZGVwYWdlICk7CgogICAgICAgICAgICBjcGluZm8tPkNvZGVQYWdlID0gdGFibGUtPmluZm8uY29kZXBhZ2U7CiAgICAgICAgICAgIGNwaW5mby0+VW5pY29kZURlZmF1bHRDaGFyID0gdGFibGUtPmluZm8uZGVmX3VuaWNvZGVfY2hhcjsKICAgICAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhciggQ1BfQUNQLCAwLCB0YWJsZS0+aW5mby5uYW1lLCAtMSwgY3BpbmZvLT5Db2RlUGFnZU5hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihjcGluZm8tPkNvZGVQYWdlTmFtZSkvc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEVudW1TeXN0ZW1Db2RlUGFnZXNBICAgKEtFUk5FTDMyLkApCiAqCiAqIENhbGwgYSB1c2VyIGRlZmluZWQgZnVuY3Rpb24gZm9yIGV2ZXJ5IGNvZGUgcGFnZSBpbnN0YWxsZWQgb24gdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICAgbHBmbkNvZGVQYWdlRW51bSBbSV0gVXNlciBDT0RFUEFHRV9FTlVNUFJPQyB0byBjYWxsIHdpdGggZWFjaCBmb3VuZCBjb2RlIHBhZ2UKICogICBmbGFncyAgICAgICAgICAgIFtJXSBSZXNlcnZlZCwgc2V0IHRvIDAuCiAqCiAqIFJFVFVSTlMKICogIFRSVUUsIElmIGFsbCBjb2RlIHBhZ2VzIGhhdmUgYmVlbiBlbnVtZXJhdGVkLCBvcgogKiAgRkFMU0UgaWYgbHBmbkNvZGVQYWdlRW51bSByZXR1cm5lZCBGQUxTRSB0byBzdG9wIHRoZSBlbnVtZXJhdGlvbi4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Db2RlUGFnZXNBKCBDT0RFUEFHRV9FTlVNUFJPQ0EgbHBmbkNvZGVQYWdlRW51bSwgRFdPUkQgZmxhZ3MgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIGNoYXIgYnVmZmVyWzEwXTsKICAgIGludCBpbmRleCA9IDA7CgogICAgZm9yICg7OykKICAgIHsKICAgICAgICBpZiAoISh0YWJsZSA9IHdpbmVfY3BfZW51bV90YWJsZSggaW5kZXgrKyApKSkgYnJlYWs7CiAgICAgICAgc3ByaW50ZiggYnVmZmVyLCAiJWQiLCB0YWJsZS0+aW5mby5jb2RlcGFnZSApOwogICAgICAgIGlmICghbHBmbkNvZGVQYWdlRW51bSggYnVmZmVyICkpIGJyZWFrOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEVudW1TeXN0ZW1Db2RlUGFnZXNXICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBFbnVtU3lzdGVtQ29kZVBhZ2VzQS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Db2RlUGFnZXNXKCBDT0RFUEFHRV9FTlVNUFJPQ1cgbHBmbkNvZGVQYWdlRW51bSwgRFdPUkQgZmxhZ3MgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIFdDSEFSIGJ1ZmZlclsxMF0sICpwOwogICAgaW50IHBhZ2UsIGluZGV4ID0gMDsKCiAgICBmb3IgKDs7KQogICAgewogICAgICAgIGlmICghKHRhYmxlID0gd2luZV9jcF9lbnVtX3RhYmxlKCBpbmRleCsrICkpKSBicmVhazsKICAgICAgICBwID0gYnVmZmVyICsgc2l6ZW9mKGJ1ZmZlcikvc2l6ZW9mKFdDSEFSKTsKICAgICAgICAqLS1wID0gMDsKICAgICAgICBwYWdlID0gdGFibGUtPmluZm8uY29kZXBhZ2U7CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICAgICotLXAgPSAnMCcgKyAocGFnZSAlIDEwKTsKICAgICAgICAgICAgcGFnZSAvPSAxMDsKICAgICAgICB9IHdoaWxlKCBwYWdlICk7CiAgICAgICAgaWYgKCFscGZuQ29kZVBhZ2VFbnVtKCBwICkpIGJyZWFrOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIgICAoS0VSTkVMMzIuQCkKICoKICogQ29udmVydCBhIG11bHRpYnl0ZSBjaGFyYWN0ZXIgc3RyaW5nIGludG8gYSBVbmljb2RlIHN0cmluZy4KICoKICogUEFSQU1TCiAqICAgcGFnZSAgIFtJXSBDb2RlcGFnZSBjaGFyYWN0ZXIgc2V0IHRvIGNvbnZlcnQgZnJvbQogKiAgIGZsYWdzICBbSV0gQ2hhcmFjdGVyIG1hcHBpbmcgZmxhZ3MKICogICBzcmMgICAgW0ldIFNvdXJjZSBzdHJpbmcgYnVmZmVyCiAqICAgc3JjbGVuIFtJXSBMZW5ndGggb2Ygc3JjLCBvciAtMSBpZiBzcmMgaXMgTlVMIHRlcm1pbmF0ZWQKICogICBkc3QgICAgW09dIERlc3RpbmF0aW9uIGJ1ZmZlcgogKiAgIGRzdGxlbiBbSV0gTGVuZ3RoIG9mIGRzdCwgb3IgMCB0byBjb21wdXRlIHRoZSByZXF1aXJlZCBsZW5ndGgKICoKICogUkVUVVJOUwogKiAgIFN1Y2Nlc3M6IElmIGRzdGxlbiA+IDAsIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyB3cml0dGVuIHRvIGRzdC4KICogICAgICAgICAgICBJZiBkc3RsZW4gPT0gMCwgdGhlIG51bWJlciBvZiBjaGFyYWN0ZXJzIG5lZWRlZCB0byBwZXJmb3JtIHRoZQogKiAgICAgICAgICAgIGNvbnZlcnNpb24uIEluIGJvdGggY2FzZXMgdGhlIGNvdW50IGluY2x1ZGVzIHRoZSB0ZXJtaW5hdGluZyBOVUwuCiAqICAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuIFBvc3NpYmxlIGVycm9ycyBhcmUKICogICAgICAgICAgICBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSLCBpZiBub3QgZW5vdWdoIHNwYWNlIGlzIGF2YWlsYWJsZSBpbiBkc3QKICogICAgICAgICAgICBhbmQgZHN0bGVuICE9IDA7IEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSLCAgaWYgYW4gaW52YWxpZCBwYXJhbWV0ZXIKICogICAgICAgICAgICBpcyBwYXNzZWQsIGFuZCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OIGlmIG5vIHRyYW5zbGF0aW9uIGlzCiAqICAgICAgICAgICAgcG9zc2libGUgZm9yIHNyYy4KICovCklOVCBXSU5BUEkgTXVsdGlCeXRlVG9XaWRlQ2hhciggVUlOVCBwYWdlLCBEV09SRCBmbGFncywgTFBDU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUFdTVFIgZHN0LCBJTlQgZHN0bGVuICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CiAgICBpbnQgcmV0OwoKICAgIGlmICghc3JjIHx8ICghZHN0ICYmIGRzdGxlbikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzcmNsZW4gPCAwKSBzcmNsZW4gPSBzdHJsZW4oc3JjKSArIDE7CgogICAgaWYgKGZsYWdzICYgTUJfVVNFR0xZUEhDSEFSUykgRklYTUUoIk1CX1VTRUdMWVBIQ0hBUlMgbm90IHN1cHBvcnRlZFxuIik7CgogICAgc3dpdGNoKHBhZ2UpCiAgICB7CiAgICBjYXNlIENQX1NZTUJPTDoKICAgICAgICBpZiggZmxhZ3MpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KICAgICAgICByZXQgPSB3aW5lX2Nwc3ltYm9sX21ic3Rvd2NzKCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4gKTsKICAgICAgICBicmVhazsKICAgIGNhc2UgQ1BfVVRGNzoKICAgICAgICBGSVhNRSgiVVRGLTcgbm90IHN1cHBvcnRlZFxuIik7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9DQUxMX05PVF9JTVBMRU1FTlRFRCApOwogICAgICAgIHJldHVybiAwOwogICAgY2FzZSBDUF9VTklYQ1A6CiAgICAgICAgaWYgKHVuaXhfY3B0YWJsZSkKICAgICAgICB7CiAgICAgICAgICAgIHJldCA9IHdpbmVfY3BfbWJzdG93Y3MoIHVuaXhfY3B0YWJsZSwgZmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgcmV0ID0gd2luZV91dGY4X21ic3Rvd2NzKCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIGlmICghKHRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBwYWdlICkpKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9jcF9tYnN0b3djcyggdGFibGUsIGZsYWdzLCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4gKTsKICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAocmV0IDwgMCkKICAgIHsKICAgICAgICBzd2l0Y2gocmV0KQogICAgICAgIHsKICAgICAgICBjYXNlIC0xOiBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsgYnJlYWs7CiAgICAgICAgY2FzZSAtMjogU2V0TGFzdEVycm9yKCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OICk7IGJyZWFrOwogICAgICAgIH0KICAgICAgICByZXQgPSAwOwogICAgfQogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZSAgIChLRVJORUwzMi5AKQogKgogKiBDb252ZXJ0IGEgVW5pY29kZSBjaGFyYWN0ZXIgc3RyaW5nIGludG8gYSBtdWx0aWJ5dGUgc3RyaW5nLgogKgogKiBQQVJBTVMKICogICBwYWdlICAgIFtJXSBDb2RlIHBhZ2UgY2hhcmFjdGVyIHNldCB0byBjb252ZXJ0IHRvCiAqICAgZmxhZ3MgICBbSV0gTWFwcGluZyBGbGFncyAoTUJfIGNvbnN0YW50cyBmcm9tICJ3aW5ubHMuaCIpLgogKiAgIHNyYyAgICAgW0ldIFNvdXJjZSBzdHJpbmcgYnVmZmVyCiAqICAgc3JjbGVuICBbSV0gTGVuZ3RoIG9mIHNyYywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICAgZHN0ICAgICBbT10gRGVzdGluYXRpb24gYnVmZmVyCiAqICAgZHN0bGVuICBbSV0gTGVuZ3RoIG9mIGRzdCwgb3IgMCB0byBjb21wdXRlIHRoZSByZXF1aXJlZCBsZW5ndGgKICogICBkZWZjaGFyIFtJXSBEZWZhdWx0IGNoYXJhY3RlciB0byB1c2UgZm9yIGNvbnZlcnNpb24gaWYgbm8gZXhhY3QKICoJCSAgICBjb252ZXJzaW9uIGNhbiBiZSBtYWRlCiAqICAgdXNlZCAgICBbT10gU2V0IGlmIGRlZmF1bHQgY2hhcmFjdGVyIHdhcyB1c2VkIGluIHRoZSBjb252ZXJzaW9uCiAqCiAqIFJFVFVSTlMKICogICBTdWNjZXNzOiBJZiBkc3RsZW4gPiAwLCB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgd3JpdHRlbiB0byBkc3QuCiAqICAgICAgICAgICAgSWYgZHN0bGVuID09IDAsIG51bWJlciBvZiBjaGFyYWN0ZXJzIG5lZWRlZCB0byBwZXJmb3JtIHRoZQogKiAgICAgICAgICAgIGNvbnZlcnNpb24uIEluIGJvdGggY2FzZXMgdGhlIGNvdW50IGluY2x1ZGVzIHRoZSB0ZXJtaW5hdGluZyBOVUwuCiAqICAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuIFBvc3NpYmxlIGVycm9ycyBhcmUKICogICAgICAgICAgICBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSLCBpZiBub3QgZW5vdWdoIHNwYWNlIGlzIGF2YWlsYWJsZSBpbiBkc3QKICogICAgICAgICAgICBhbmQgZHN0bGVuICE9IDAsIGFuZCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiwgaWYgYW4gaW52YWxpZAogKiAgICAgICAgICAgIHBhcmFtZXRlciB3YXMgZ2l2ZW4uCiAqLwpJTlQgV0lOQVBJIFdpZGVDaGFyVG9NdWx0aUJ5dGUoIFVJTlQgcGFnZSwgRFdPUkQgZmxhZ3MsIExQQ1dTVFIgc3JjLCBJTlQgc3JjbGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQU1RSIGRzdCwgSU5UIGRzdGxlbiwgTFBDU1RSIGRlZmNoYXIsIEJPT0wgKnVzZWQgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIGludCByZXQsIHVzZWRfdG1wOwoKICAgIGlmICghc3JjIHx8ICghZHN0ICYmIGRzdGxlbikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzcmNsZW4gPCAwKSBzcmNsZW4gPSBzdHJsZW5XKHNyYykgKyAxOwoKICAgIHN3aXRjaChwYWdlKQogICAgewogICAgY2FzZSBDUF9TWU1CT0w6CiAgICAgICAgaWYoIGZsYWdzIHx8IGRlZmNoYXIgfHwgdXNlZCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfY3BzeW1ib2xfd2NzdG9tYnMoIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBDUF9VVEY3OgogICAgICAgIEZJWE1FKCJVVEYtNyBub3Qgc3VwcG9ydGVkXG4iKTsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0NBTExfTk9UX0lNUExFTUVOVEVEICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICBjYXNlIENQX1VOSVhDUDoKICAgICAgICBpZiAodW5peF9jcHRhYmxlKQogICAgICAgIHsKICAgICAgICAgICAgcmV0ID0gd2luZV9jcF93Y3N0b21icyggdW5peF9jcHRhYmxlLCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZWZjaGFyLCB1c2VkID8gJnVzZWRfdG1wIDogTlVMTCApOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgaWYgKHVzZWQpICp1c2VkID0gRkFMU0U7ICAvKiBhbGwgY2hhcnMgYXJlIHZhbGlkIGZvciBVVEYtOCAqLwogICAgICAgIHJldCA9IHdpbmVfdXRmOF93Y3N0b21icyggc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIGlmICghKHRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBwYWdlICkpKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9jcF93Y3N0b21icyggdGFibGUsIGZsYWdzLCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVmY2hhciwgdXNlZCA/ICZ1c2VkX3RtcCA6IE5VTEwgKTsKICAgICAgICBpZiAodXNlZCkgKnVzZWQgPSB1c2VkX3RtcDsKICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAocmV0IDwgMCkKICAgIHsKICAgICAgICBzd2l0Y2gocmV0KQogICAgICAgIHsKICAgICAgICBjYXNlIC0xOiBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsgYnJlYWs7CiAgICAgICAgY2FzZSAtMjogU2V0TGFzdEVycm9yKCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OICk7IGJyZWFrOwogICAgICAgIH0KICAgICAgICByZXQgPSAwOwogICAgfQogICAgVFJBQ0UoImNwICVkICVzIC0+ICVzXG4iLCBwYWdlLCBkZWJ1Z3N0cl93KHNyYyksIGRlYnVnc3RyX2EoZHN0KSk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRUaHJlYWRMb2NhbGUgICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgY3VycmVudCB0aHJlYWRzIGxvY2FsZS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgTENJRCBjdXJyZW50bHkgYXNzb2NhdGVkIHdpdGggdGhlIGNhbGxpbmcgdGhyZWFkLgogKi8KTENJRCBXSU5BUEkgR2V0VGhyZWFkTG9jYWxlKHZvaWQpCnsKICAgIExDSUQgcmV0ID0gTnRDdXJyZW50VGViKCktPkN1cnJlbnRMb2NhbGU7CiAgICBpZiAoIXJldCkgTnRDdXJyZW50VGViKCktPkN1cnJlbnRMb2NhbGUgPSByZXQgPSBHZXRVc2VyRGVmYXVsdExDSUQoKTsKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBTZXRUaHJlYWRMb2NhbGUgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNldCB0aGUgY3VycmVudCB0aHJlYWRzIGxvY2FsZS4KICoKICogUEFSQU1TCiAqICBsY2lkIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUgdG8gc2V0CiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIFRoZSB0aHJlYWRzIGxvY2FsZSBpcyBzZXQgdG8gbGNpZC4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIFNldFRocmVhZExvY2FsZSggTENJRCBsY2lkICkKewogICAgVFJBQ0UoIigweCUwNGxYKVxuIiwgbGNpZCk7CgogICAgbGNpZCA9IENvbnZlcnREZWZhdWx0TG9jYWxlKGxjaWQpOwoKICAgIGlmIChsY2lkICE9IEdldFRocmVhZExvY2FsZSgpKQogICAgewogICAgICAgIGlmICghSXNWYWxpZExvY2FsZShsY2lkLCBMQ0lEX1NVUFBPUlRFRCkpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgICAgICByZXR1cm4gRkFMU0U7CiAgICAgICAgfQoKICAgICAgICBOdEN1cnJlbnRUZWIoKS0+Q3VycmVudExvY2FsZSA9IGxjaWQ7CiAgICAgICAga2VybmVsX2dldF90aHJlYWRfZGF0YSgpLT5jb2RlX3BhZ2UgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUNvbnZlcnREZWZhdWx0TG9jYWxlIChLRVJORUwzMi5AKQogKgogKiBDb252ZXJ0IGEgZGVmYXVsdCBsb2NhbGUgaWRlbnRpZmllciBpbnRvIGEgcmVhbCBpZGVudGlmaWVyLgogKgogKiBQQVJBTVMKICogIGxjaWQgW0ldIExDSUQgaWRlbnRpZmllciBvZiB0aGUgbG9jYWxlIHRvIGNvbnZlcnQKICoKICogUkVUVVJOUwogKiAgbGNpZCB1bmNoYW5nZWQsIGlmIG5vdCBhIGRlZmF1bHQgbG9jYWxlIG9yIGl0cyBzdWJsYW5ndWFnZSBpcwogKiAgIG5vdCBTVUJMQU5HX05FVVRSQUwuCiAqICBHZXRTeXN0ZW1EZWZhdWx0TENJRCgpLCBpZiBsY2lkID09IExPQ0FMRV9TWVNURU1fREVGQVVMVC4KICogIEdldFVzZXJEZWZhdWx0TENJRCgpLCBpZiBsY2lkID09IExPQ0FMRV9VU0VSX0RFRkFVTFQgb3IgTE9DQUxFX05FVVRSQUwuCiAqICBPdGhlcndpc2UsIGxjaWQgd2l0aCBzdWJsYW5ndWFnZSBjaGFuZ2VkIHRvIFNVQkxBTkdfREVGQVVMVC4KICovCkxDSUQgV0lOQVBJIENvbnZlcnREZWZhdWx0TG9jYWxlKCBMQ0lEIGxjaWQgKQp7CiAgICBMQU5HSUQgbGFuZ2lkOwoKICAgIHN3aXRjaCAobGNpZCkKICAgIHsKICAgIGNhc2UgTE9DQUxFX1NZU1RFTV9ERUZBVUxUOgogICAgICAgIGxjaWQgPSBHZXRTeXN0ZW1EZWZhdWx0TENJRCgpOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBMT0NBTEVfVVNFUl9ERUZBVUxUOgogICAgY2FzZSBMT0NBTEVfTkVVVFJBTDoKICAgICAgICBsY2lkID0gR2V0VXNlckRlZmF1bHRMQ0lEKCk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIC8qIFJlcGxhY2UgU1VCTEFOR19ORVVUUkFMIHdpdGggU1VCTEFOR19ERUZBVUxUICovCiAgICAgICAgbGFuZ2lkID0gTEFOR0lERlJPTUxDSUQobGNpZCk7CiAgICAgICAgaWYgKFNVQkxBTkdJRChsYW5naWQpID09IFNVQkxBTkdfTkVVVFJBTCkKICAgICAgICB7CiAgICAgICAgICBsYW5naWQgPSBNQUtFTEFOR0lEKFBSSU1BUllMQU5HSUQobGFuZ2lkKSwgU1VCTEFOR19ERUZBVUxUKTsKICAgICAgICAgIGxjaWQgPSBNQUtFTENJRChsYW5naWQsIFNPUlRJREZST01MQ0lEKGxjaWQpKTsKICAgICAgICB9CiAgICB9CiAgICByZXR1cm4gbGNpZDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIElzVmFsaWRMb2NhbGUgICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgbG9jYWxlIGlzIHZhbGlkLgogKgogKiBQQVJBTVMKICogIGxjaWQgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUgdG8gY2hlY2sKICogIGZsYWdzIFtJXSBMQ0lEX1NVUFBPUlRFRCA9IFZhbGlkLCBMQ0lEX0lOU1RBTExFRCA9IFZhbGlkIGFuZCBpbnN0YWxsZWQgb24gdGhlIHN5c3RlbQogKgogKiBSRVRVUk5TCiAqICBUUlVFLCAgaWYgbGNpZCBpcyB2YWxpZCwKICogIEZBTFNFLCBvdGhlcndpc2UuCiAqCiAqIE5PVEVTCiAqICBXaW5lIGRvZXMgbm90IGN1cnJlbnRseSBtYWtlIHRoZSBkaXN0aW5jdGlvbiBiZXR3ZWVuIHN1cHBvcnRlZCBhbmQgaW5zdGFsbGVkLiBBbGwKICogIGxhbmd1YWdlcyBzdXBwb3J0ZWQgYXJlIGluc3RhbGxlZCBieSBkZWZhdWx0LgogKi8KQk9PTCBXSU5BUEkgSXNWYWxpZExvY2FsZSggTENJRCBsY2lkLCBEV09SRCBmbGFncyApCnsKICAgIC8qIGNoZWNrIGlmIGxhbmd1YWdlIGlzIHJlZ2lzdGVyZWQgaW4gdGhlIGtlcm5lbDMyIHJlc291cmNlcyAqLwogICAgcmV0dXJuIEZpbmRSZXNvdXJjZUV4Vygga2VybmVsMzJfaGFuZGxlLCAoTFBXU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIExBTkdJREZST01MQ0lEKGxjaWQpKSAhPSAwOwp9CgoKc3RhdGljIEJPT0wgQ0FMTEJBQ0sgZW51bV9sYW5nX3Byb2NfYSggSE1PRFVMRSBoTW9kdWxlLCBMUENTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDU1RSIG5hbWUsIFdPUkQgTGFuZ0lELCBMT05HX1BUUiBsUGFyYW0gKQp7CiAgICBMT0NBTEVfRU5VTVBST0NBIGxwZm5Mb2NhbGVFbnVtID0gKExPQ0FMRV9FTlVNUFJPQ0EpbFBhcmFtOwogICAgY2hhciBidWZbMjBdOwoKICAgIHNwcmludGYoYnVmLCAiJTA4eCIsIChVSU5UKUxhbmdJRCk7CiAgICByZXR1cm4gbHBmbkxvY2FsZUVudW0oIGJ1ZiApOwp9CgpzdGF0aWMgQk9PTCBDQUxMQkFDSyBlbnVtX2xhbmdfcHJvY193KCBITU9EVUxFIGhNb2R1bGUsIExQQ1dTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDV1NUUiBuYW1lLCBXT1JEIExhbmdJRCwgTE9OR19QVFIgbFBhcmFtICkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGZvcm1hdFdbXSA9IHsnJScsJzAnLCc4JywneCcsMH07CiAgICBMT0NBTEVfRU5VTVBST0NXIGxwZm5Mb2NhbGVFbnVtID0gKExPQ0FMRV9FTlVNUFJPQ1cpbFBhcmFtOwogICAgV0NIQVIgYnVmWzIwXTsKICAgIHNwcmludGZXKCBidWYsIGZvcm1hdFcsIChVSU5UKUxhbmdJRCApOwogICAgcmV0dXJuIGxwZm5Mb2NhbGVFbnVtKCBidWYgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxvY2FsZXNBICAoS0VSTkVMMzIuQCkKICoKICogQ2FsbCBhIHVzZXJzIGZ1bmN0aW9uIGZvciBlYWNoIGxvY2FsZSBhdmFpbGFibGUgb24gdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBscGZuTG9jYWxlRW51bSBbSV0gQ2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCBmb3IgZWFjaCBsb2NhbGUKICogIGR3RmxhZ3MgICAgICAgIFtJXSBMT0NBTEVfU1VQUE9SVEVEPUFsbCBzdXBwb3J0ZWQsIExPQ0FMRV9JTlNUQUxMRUQ9SW5zdGFsbGVkIG9ubHkKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Mb2NhbGVzQSggTE9DQUxFX0VOVU1QUk9DQSBscGZuTG9jYWxlRW51bSwgRFdPUkQgZHdGbGFncyApCnsKICAgIFRSQUNFKCIoJXAsJTA4bHgpXG4iLCBscGZuTG9jYWxlRW51bSwgZHdGbGFncyk7CiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNBKCBrZXJuZWwzMl9oYW5kbGUsIChMUFNUUilSVF9TVFJJTkcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBDU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIGVudW1fbGFuZ19wcm9jX2EsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTE9OR19QVFIpbHBmbkxvY2FsZUVudW0pOwogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtTG9jYWxlc1cgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRW51bVN5c3RlbUxvY2FsZXNBLgogKi8KQk9PTCBXSU5BUEkgRW51bVN5c3RlbUxvY2FsZXNXKCBMT0NBTEVfRU5VTVBST0NXIGxwZm5Mb2NhbGVFbnVtLCBEV09SRCBkd0ZsYWdzICkKewogICAgVFJBQ0UoIiglcCwlMDhseClcbiIsIGxwZm5Mb2NhbGVFbnVtLCBkd0ZsYWdzKTsKICAgIEVudW1SZXNvdXJjZUxhbmd1YWdlc1coIGtlcm5lbDMyX2hhbmRsZSwgKExQV1NUUilSVF9TVFJJTkcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBDV1NUUilMT0NBTEVfSUxBTkdVQUdFLCBlbnVtX2xhbmdfcHJvY193LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExPTkdfUFRSKWxwZm5Mb2NhbGVFbnVtKTsKICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBWZXJMYW5ndWFnZU5hbWVBICAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBuYW1lIG9mIGEgbGFuZ3VhZ2UuCiAqCiAqIFBBUkFNUwogKiAgd0xhbmcgIFtJXSBMQU5HSUQgb2YgdGhlIGxhbmd1YWdlCiAqICBzekxhbmcgW09dIERlc3RpbmF0aW9uIGZvciB0aGUgbGFuZ3VhZ2UgbmFtZQogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgc2l6ZSBvZiB0aGUgbGFuZ3VhZ2UgbmFtZS4gSWYgc3pMYW5nIGlzIG5vbi1OVUxMLCBpdCBpcyBmaWxsZWQKICogICAgICAgICAgIHdpdGggdGhlIG5hbWUuCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICovCkRXT1JEIFdJTkFQSSBWZXJMYW5ndWFnZU5hbWVBKCBVSU5UIHdMYW5nLCBMUFNUUiBzekxhbmcsIFVJTlQgblNpemUgKQp7CiAgICByZXR1cm4gR2V0TG9jYWxlSW5mb0EoIE1BS0VMQ0lEKHdMYW5nLCBTT1JUX0RFRkFVTFQpLCBMT0NBTEVfU0VOR0xBTkdVQUdFLCBzekxhbmcsIG5TaXplICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIFZlckxhbmd1YWdlTmFtZVcgIChLRVJORUwzMi5AKQogKgogKiBTZWUgVmVyTGFuZ3VhZ2VOYW1lQS4KICovCkRXT1JEIFdJTkFQSSBWZXJMYW5ndWFnZU5hbWVXKCBVSU5UIHdMYW5nLCBMUFdTVFIgc3pMYW5nLCBVSU5UIG5TaXplICkKewogICAgcmV0dXJuIEdldExvY2FsZUluZm9XKCBNQUtFTENJRCh3TGFuZywgU09SVF9ERUZBVUxUKSwgTE9DQUxFX1NFTkdMQU5HVUFHRSwgc3pMYW5nLCBuU2l6ZSApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0U3RyaW5nVHlwZVcgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBHZXRTdHJpbmdUeXBlQS4KICovCkJPT0wgV0lOQVBJIEdldFN0cmluZ1R5cGVXKCBEV09SRCB0eXBlLCBMUENXU1RSIHNyYywgSU5UIGNvdW50LCBMUFdPUkQgY2hhcnR5cGUgKQp7CiAgICBpZiAoY291bnQgPT0gLTEpIGNvdW50ID0gc3RybGVuVyhzcmMpICsgMTsKICAgIHN3aXRjaCh0eXBlKQogICAgewogICAgY2FzZSBDVF9DVFlQRTE6CiAgICAgICAgd2hpbGUgKGNvdW50LS0pICpjaGFydHlwZSsrID0gZ2V0X2NoYXJfdHlwZVcoICpzcmMrKyApICYgMHhmZmY7CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENUX0NUWVBFMjoKICAgICAgICB3aGlsZSAoY291bnQtLSkgKmNoYXJ0eXBlKysgPSBnZXRfY2hhcl90eXBlVyggKnNyYysrICkgPj4gMTI7CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENUX0NUWVBFMzoKICAgIHsKICAgICAgICBXQVJOKCJDVF9DVFlQRTM6IHNlbWktc3R1Yi5cbiIpOwogICAgICAgIHdoaWxlIChjb3VudC0tKQogICAgICAgIHsKICAgICAgICAgICAgaW50IGMgPSAqc3JjOwogICAgICAgICAgICBXT1JEIHR5cGUxLCB0eXBlMyA9IDA7IC8qIEMzX05PVEFQUExJQ0FCTEUgKi8KCiAgICAgICAgICAgIHR5cGUxID0gZ2V0X2NoYXJfdHlwZVcoICpzcmMrKyApICYgMHhmZmY7CiAgICAgICAgICAgIC8qIHRyeSB0byBjb25zdHJ1Y3QgdHlwZTMgZnJvbSB0eXBlMSAqLwogICAgICAgICAgICBpZih0eXBlMSAmIEMxX1NQQUNFKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgIGlmKHR5cGUxICYgQzFfQUxQSEEpIHR5cGUzIHw9IEMzX0FMUEhBOwogICAgICAgICAgICBpZiAoKGM+PTB4MzBBMCkmJihjPD0weDMwRkYpKSB0eXBlMyB8PSBDM19LQVRBS0FOQTsKICAgICAgICAgICAgaWYgKChjPj0weDMwNDApJiYoYzw9MHgzMDlGKSkgdHlwZTMgfD0gQzNfSElSQUdBTkE7CiAgICAgICAgICAgIGlmICgoYz49MHg0RTAwKSYmKGM8PTB4OUZBRikpIHR5cGUzIHw9IEMzX0lERU9HUkFQSDsKICAgICAgICAgICAgaWYgKChjPj0weDA2MDApJiYoYzw9MHgwNkZGKSkgdHlwZTMgfD0gQzNfS0FTSElEQTsKICAgICAgICAgICAgaWYgKChjPj0weDMwMDApJiYoYzw9MHgzMDNGKSkgdHlwZTMgfD0gQzNfU1lNQk9MOwoKICAgICAgICAgICAgaWYgKChjPj0weEZGMDApJiYoYzw9MHhGRjYwKSkgdHlwZTMgfD0gQzNfRlVMTFdJRFRIOwogICAgICAgICAgICBpZiAoKGM+PTB4RkYwMCkmJihjPD0weEZGMjApKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjNCKSYmKGM8PTB4RkY0MCkpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgaWYgKChjPj0weEZGNUIpJiYoYzw9MHhGRjYwKSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICBpZiAoKGM+PTB4RkYyMSkmJihjPD0weEZGM0EpKSB0eXBlMyB8PSBDM19BTFBIQTsKICAgICAgICAgICAgaWYgKChjPj0weEZGNDEpJiYoYzw9MHhGRjVBKSkgdHlwZTMgfD0gQzNfQUxQSEE7CiAgICAgICAgICAgIGlmICgoYz49MHhGRkUwKSYmKGM8PTB4RkZFNikpIHR5cGUzIHw9IEMzX0ZVTExXSURUSDsKICAgICAgICAgICAgaWYgKChjPj0weEZGRTApJiYoYzw9MHhGRkU2KSkgdHlwZTMgfD0gQzNfU1lNQk9MOwoKICAgICAgICAgICAgaWYgKChjPj0weEZGNjEpJiYoYzw9MHhGRkRDKSkgdHlwZTMgfD0gQzNfSEFMRldJRFRIOwogICAgICAgICAgICBpZiAoKGM+PTB4RkY2MSkmJihjPD0weEZGNjQpKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjY1KSYmKGM8PTB4RkY5RikpIHR5cGUzIHw9IEMzX0tBVEFLQU5BOwogICAgICAgICAgICBpZiAoKGM+PTB4RkY2NSkmJihjPD0weEZGOUYpKSB0eXBlMyB8PSBDM19BTFBIQTsKICAgICAgICAgICAgaWYgKChjPj0weEZGRTgpJiYoYzw9MHhGRkVFKSkgdHlwZTMgfD0gQzNfSEFMRldJRFRIOwogICAgICAgICAgICBpZiAoKGM+PTB4RkZFOCkmJihjPD0weEZGRUUpKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgICpjaGFydHlwZSsrID0gdHlwZTM7CiAgICAgICAgfQogICAgICAgIGJyZWFrOwogICAgfQogICAgZGVmYXVsdDoKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRTdHJpbmdUeXBlRXhXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgR2V0U3RyaW5nVHlwZUV4QS4KICovCkJPT0wgV0lOQVBJIEdldFN0cmluZ1R5cGVFeFcoIExDSUQgbG9jYWxlLCBEV09SRCB0eXBlLCBMUENXU1RSIHNyYywgSU5UIGNvdW50LCBMUFdPUkQgY2hhcnR5cGUgKQp7CiAgICAvKiBsb2NhbGUgaXMgaWdub3JlZCBmb3IgVW5pY29kZSAqLwogICAgcmV0dXJuIEdldFN0cmluZ1R5cGVXKCB0eXBlLCBzcmMsIGNvdW50LCBjaGFydHlwZSApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0U3RyaW5nVHlwZUEgICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBjaGFyYWN0ZXJpc3RpY3Mgb2YgdGhlIGNoYXJhY3RlcnMgbWFraW5nIHVwIGEgc3RyaW5nLgogKgogKiBQQVJBTVMKICogIGxvY2FsZSAgIFtJXSBMb2NhbGUgSWQgZm9yIHRoZSBzdHJpbmcKICogIHR5cGUgICAgIFtJXSBDVF9DVFlQRTEgPSBjbGFzc2lmaWNhdGlvbiwgQ1RfQ1RZUEUyID0gZGlyZWN0aW9uYWxpdHksIENUX0NUWVBFMyA9IHR5cG9ncmFwaGljIGluZm8KICogIHNyYyAgICAgIFtJXSBTdHJpbmcgdG8gYW5hbHlzZQogKiAgY291bnQgICAgW0ldIExlbmd0aCBvZiBzcmMgaW4gY2hhcnMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgY2hhcnR5cGUgW09dIERlc3RpbmF0aW9uIGZvciB0aGUgY2FsY3VsYXRlZCBjaGFyYWN0ZXJpc3RpY3MKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4gY2hhcnR5cGUgaXMgZmlsbGVkIHdpdGggdGhlIHJlcXVlc3RlZCBjaGFyYWN0ZXJpc3RpY3Mgb2YgZWFjaCBjaGFyCiAqICAgICAgICAgICBpbiBzcmMuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBHZXRTdHJpbmdUeXBlQSggTENJRCBsb2NhbGUsIERXT1JEIHR5cGUsIExQQ1NUUiBzcmMsIElOVCBjb3VudCwgTFBXT1JEIGNoYXJ0eXBlICkKewogICAgVUlOVCBjcDsKICAgIElOVCBjb3VudFc7CiAgICBMUFdTVFIgc3JjVzsKICAgIEJPT0wgcmV0ID0gRkFMU0U7CgogICAgaWYoY291bnQgPT0gLTEpIGNvdW50ID0gc3RybGVuKHNyYykgKyAxOwoKICAgIGlmICghKGNwID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxvY2FsZSApKSkKICAgIHsKICAgICAgICBGSVhNRSgiRm9yIGxvY2FsZSAlMDRseCB1c2luZyBjdXJyZW50IEFOU0kgY29kZSBwYWdlXG4iLCBsb2NhbGUpOwogICAgICAgIGNwID0gR2V0QUNQKCk7CiAgICB9CgogICAgY291bnRXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihjcCwgMCwgc3JjLCBjb3VudCwgTlVMTCwgMCk7CiAgICBpZigoc3JjVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBjb3VudFcgKiBzaXplb2YoV0NIQVIpKSkpCiAgICB7CiAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihjcCwgMCwgc3JjLCBjb3VudCwgc3JjVywgY291bnRXKTsKICAgIC8qCiAgICAgKiBOT1RFOiB0aGUgdGFyZ2V0IGJ1ZmZlciBoYXMgMSB3b3JkIGZvciBlYWNoIENIQVJBQ1RFUiBpbiB0aGUgc291cmNlCiAgICAgKiBzdHJpbmcsIHdpdGggbXVsdGlieXRlIGNoYXJhY3RlcnMgdGhlcmUgbWF5YmUgYmUgbW9yZSBieXRlcyBpbiBjb3VudAogICAgICogdGhhbiBjaGFyYWN0ZXIgc3BhY2UgaW4gdGhlIGJ1ZmZlciEKICAgICAqLwogICAgICAgIHJldCA9IEdldFN0cmluZ1R5cGVXKHR5cGUsIHNyY1csIGNvdW50VywgY2hhcnR5cGUpOwogICAgICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgfQogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0U3RyaW5nVHlwZUV4QSAgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGNoYXJhY3RlcmlzdGljcyBvZiB0aGUgY2hhcmFjdGVycyBtYWtpbmcgdXAgYSBzdHJpbmcuCiAqCiAqIFBBUkFNUwogKiAgbG9jYWxlICAgW0ldIExvY2FsZSBJZCBmb3IgdGhlIHN0cmluZwogKiAgdHlwZSAgICAgW0ldIENUX0NUWVBFMSA9IGNsYXNzaWZpY2F0aW9uLCBDVF9DVFlQRTIgPSBkaXJlY3Rpb25hbGl0eSwgQ1RfQ1RZUEUzID0gdHlwb2dyYXBoaWMgaW5mbwogKiAgc3JjICAgICAgW0ldIFN0cmluZyB0byBhbmFseXNlCiAqICBjb3VudCAgICBbSV0gTGVuZ3RoIG9mIHNyYyBpbiBjaGFycywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBjaGFydHlwZSBbT10gRGVzdGluYXRpb24gZm9yIHRoZSBjYWxjdWxhdGVkIGNoYXJhY3RlcmlzdGljcwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBjaGFydHlwZSBpcyBmaWxsZWQgd2l0aCB0aGUgcmVxdWVzdGVkIGNoYXJhY3RlcmlzdGljcyBvZiBlYWNoIGNoYXIKICogICAgICAgICAgIGluIHNyYy4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEdldFN0cmluZ1R5cGVFeEEoIExDSUQgbG9jYWxlLCBEV09SRCB0eXBlLCBMUENTVFIgc3JjLCBJTlQgY291bnQsIExQV09SRCBjaGFydHlwZSApCnsKICAgIHJldHVybiBHZXRTdHJpbmdUeXBlQShsb2NhbGUsIHR5cGUsIHNyYywgY291bnQsIGNoYXJ0eXBlKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBMQ01hcFN0cmluZ1cgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBMQ01hcFN0cmluZ0EuCiAqLwpJTlQgV0lOQVBJIExDTWFwU3RyaW5nVyhMQ0lEIGxjaWQsIERXT1JEIGZsYWdzLCBMUENXU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICAgTFBXU1RSIGRzdCwgSU5UIGRzdGxlbikKewogICAgTFBXU1RSIGRzdF9wdHI7CgogICAgaWYgKCFzcmMgfHwgIXNyY2xlbiB8fCBkc3RsZW4gPCAwKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgLyogbXV0dWFsbHkgZXhjbHVzaXZlIGZsYWdzICovCiAgICBpZiAoKGZsYWdzICYgKExDTUFQX0xPV0VSQ0FTRSB8IExDTUFQX1VQUEVSQ0FTRSkpID09IChMQ01BUF9MT1dFUkNBU0UgfCBMQ01BUF9VUFBFUkNBU0UpIHx8CiAgICAgICAgKGZsYWdzICYgKExDTUFQX0hJUkFHQU5BIHwgTENNQVBfS0FUQUtBTkEpKSA9PSAoTENNQVBfSElSQUdBTkEgfCBMQ01BUF9LQVRBS0FOQSkgfHwKICAgICAgICAoZmxhZ3MgJiAoTENNQVBfSEFMRldJRFRIIHwgTENNQVBfRlVMTFdJRFRIKSkgPT0gKExDTUFQX0hBTEZXSURUSCB8IExDTUFQX0ZVTExXSURUSCkgfHwKICAgICAgICAoZmxhZ3MgJiAoTENNQVBfVFJBRElUSU9OQUxfQ0hJTkVTRSB8IExDTUFQX1NJTVBMSUZJRURfQ0hJTkVTRSkpID09IChMQ01BUF9UUkFESVRJT05BTF9DSElORVNFIHwgTENNQVBfU0lNUExJRklFRF9DSElORVNFKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKCFkc3RsZW4pIGRzdCA9IE5VTEw7CgogICAgbGNpZCA9IENvbnZlcnREZWZhdWx0TG9jYWxlKGxjaWQpOwoKICAgIGlmIChmbGFncyAmIExDTUFQX1NPUlRLRVkpCiAgICB7CiAgICAgICAgaWYgKHNyYyA9PSBkc3QpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KCiAgICAgICAgaWYgKHNyY2xlbiA8IDApIHNyY2xlbiA9IHN0cmxlblcoc3JjKTsKCiAgICAgICAgVFJBQ0UoIigweCUwNGx4LDB4JTA4bHgsJXMsJWQsJXAsJWQpXG4iLAogICAgICAgICAgICAgIGxjaWQsIGZsYWdzLCBkZWJ1Z3N0cl93bihzcmMsIHNyY2xlbiksIHNyY2xlbiwgZHN0LCBkc3RsZW4pOwoKICAgICAgICByZXR1cm4gd2luZV9nZXRfc29ydGtleShmbGFncywgc3JjLCBzcmNsZW4sIChjaGFyICopZHN0LCBkc3RsZW4pOwogICAgfQoKICAgIC8qIFNPUlRfU1RSSU5HU09SVCBtdXN0IGJlIHVzZWQgZXhjbHVzaXZlbHkgd2l0aCBMQ01BUF9TT1JUS0VZICovCiAgICBpZiAoZmxhZ3MgJiBTT1JUX1NUUklOR1NPUlQpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzcmNsZW4gPCAwKSBzcmNsZW4gPSBzdHJsZW5XKHNyYykgKyAxOwoKICAgIFRSQUNFKCIoMHglMDRseCwweCUwOGx4LCVzLCVkLCVwLCVkKVxuIiwKICAgICAgICAgIGxjaWQsIGZsYWdzLCBkZWJ1Z3N0cl93bihzcmMsIHNyY2xlbiksIHNyY2xlbiwgZHN0LCBkc3RsZW4pOwoKICAgIGlmICghZHN0KSAvKiByZXR1cm4gcmVxdWlyZWQgc3RyaW5nIGxlbmd0aCAqLwogICAgewogICAgICAgIElOVCBsZW47CgogICAgICAgIGZvciAobGVuID0gMDsgc3JjbGVuOyBzcmMrKywgc3JjbGVuLS0pCiAgICAgICAgewogICAgICAgICAgICBXQ0hBUiB3Y2ggPSAqc3JjOwogICAgICAgICAgICAvKiB0ZXN0cyBzaG93IHRoYXQgd2luMmsganVzdCBpZ25vcmVzIE5PUk1fSUdOT1JFTk9OU1BBQ0UsCiAgICAgICAgICAgICAqIGFuZCBza2lwcyB3aGl0ZSBzcGFjZSBhbmQgcHVuY3R1YXRpb24gY2hhcmFjdGVycyBmb3IKICAgICAgICAgICAgICogTk9STV9JR05PUkVTWU1CT0xTLgogICAgICAgICAgICAgKi8KICAgICAgICAgICAgaWYgKChmbGFncyAmIE5PUk1fSUdOT1JFU1lNQk9MUykgJiYgKGdldF9jaGFyX3R5cGVXKHdjaCkgJiAoQzFfUFVOQ1QgfCBDMV9TUEFDRSkpKQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgIGxlbisrOwogICAgICAgIH0KICAgICAgICByZXR1cm4gbGVuOwogICAgfQoKICAgIGlmIChmbGFncyAmIExDTUFQX1VQUEVSQ0FTRSkKICAgIHsKICAgICAgICBmb3IgKGRzdF9wdHIgPSBkc3Q7IHNyY2xlbiAmJiBkc3RsZW47IHNyYysrLCBzcmNsZW4tLSkKICAgICAgICB7CiAgICAgICAgICAgIFdDSEFSIHdjaCA9ICpzcmM7CiAgICAgICAgICAgIGlmICgoZmxhZ3MgJiBOT1JNX0lHTk9SRVNZTUJPTFMpICYmIChnZXRfY2hhcl90eXBlVyh3Y2gpICYgKEMxX1BVTkNUIHwgQzFfU1BBQ0UpKSkKICAgICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgICAqZHN0X3B0cisrID0gdG91cHBlclcod2NoKTsKICAgICAgICAgICAgZHN0bGVuLS07CiAgICAgICAgfQogICAgfQogICAgZWxzZSBpZiAoZmxhZ3MgJiBMQ01BUF9MT1dFUkNBU0UpCiAgICB7CiAgICAgICAgZm9yIChkc3RfcHRyID0gZHN0OyBzcmNsZW4gJiYgZHN0bGVuOyBzcmMrKywgc3JjbGVuLS0pCiAgICAgICAgewogICAgICAgICAgICBXQ0hBUiB3Y2ggPSAqc3JjOwogICAgICAgICAgICBpZiAoKGZsYWdzICYgTk9STV9JR05PUkVTWU1CT0xTKSAmJiAoZ2V0X2NoYXJfdHlwZVcod2NoKSAmIChDMV9QVU5DVCB8IEMxX1NQQUNFKSkpCiAgICAgICAgICAgICAgICBjb250aW51ZTsKICAgICAgICAgICAgKmRzdF9wdHIrKyA9IHRvbG93ZXJXKHdjaCk7CiAgICAgICAgICAgIGRzdGxlbi0tOwogICAgICAgIH0KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICBpZiAoc3JjID09IGRzdCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIGZvciAoZHN0X3B0ciA9IGRzdDsgc3JjbGVuICYmIGRzdGxlbjsgc3JjKyssIHNyY2xlbi0tKQogICAgICAgIHsKICAgICAgICAgICAgV0NIQVIgd2NoID0gKnNyYzsKICAgICAgICAgICAgaWYgKChmbGFncyAmIE5PUk1fSUdOT1JFU1lNQk9MUykgJiYgKGdldF9jaGFyX3R5cGVXKHdjaCkgJiAoQzFfUFVOQ1QgfCBDMV9TUEFDRSkpKQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgICpkc3RfcHRyKysgPSB3Y2g7CiAgICAgICAgICAgIGRzdGxlbi0tOwogICAgICAgIH0KICAgIH0KCiAgICBpZiAoc3JjbGVuKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICByZXR1cm4gZHN0X3B0ciAtIGRzdDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIExDTWFwU3RyaW5nQSAgICAoS0VSTkVMMzIuQCkKICoKICogTWFwIGNoYXJhY3RlcnMgaW4gYSBsb2NhbGUgc2Vuc2l0aXZlIHN0cmluZy4KICoKICogUEFSQU1TCiAqICBsY2lkICAgW0ldIExDSUQgZm9yIHRoZSBjb252ZXJzaW9uLgogKiAgZmxhZ3MgIFtJXSBGbGFncyBjb250cm9sbGluZyB0aGUgbWFwcGluZyAoTENNQVBfIGNvbnN0YW50cyBmcm9tICJ3aW5ubHMuaCIpLgogKiAgc3JjICAgIFtJXSBTdHJpbmcgdG8gbWFwCiAqICBzcmNsZW4gW0ldIExlbmd0aCBvZiBzcmMgaW4gY2hhcnMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgZHN0ICAgIFtPXSBEZXN0aW5hdGlvbiBmb3IgbWFwcGVkIHN0cmluZwogKiAgZHN0bGVuIFtJXSBMZW5ndGggb2YgZHN0IGluIGNoYXJhY3RlcnMKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVGhlIGxlbmd0aCBvZiB0aGUgbWFwcGVkIHN0cmluZyBpbiBkc3QsIGluY2x1ZGluZyB0aGUgTlVMIHRlcm1pbmF0b3IuCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCklOVCBXSU5BUEkgTENNYXBTdHJpbmdBKExDSUQgbGNpZCwgRFdPUkQgZmxhZ3MsIExQQ1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgIExQU1RSIGRzdCwgSU5UIGRzdGxlbikKewogICAgV0NIQVIgKmJ1ZlcgPSBOdEN1cnJlbnRUZWIoKS0+U3RhdGljVW5pY29kZUJ1ZmZlcjsKICAgIExQV1NUUiBzcmNXLCBkc3RXOwogICAgSU5UIHJldCA9IDAsIHNyY2xlblcsIGRzdGxlblc7CiAgICBVSU5UIGxvY2FsZV9jcCA9IENQX0FDUDsKCiAgICBpZiAoIXNyYyB8fCAhc3JjbGVuIHx8IGRzdGxlbiA8IDApCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBpZiAoIShmbGFncyAmIExPQ0FMRV9VU0VfQ1BfQUNQKSkgbG9jYWxlX2NwID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKCiAgICBzcmNsZW5XID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHNyYywgc3JjbGVuLCBidWZXLCAyNjApOwogICAgaWYgKHNyY2xlblcpCiAgICAgICAgc3JjVyA9IGJ1Zlc7CiAgICBlbHNlCiAgICB7CiAgICAgICAgc3JjbGVuVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzcmMsIHNyY2xlbiwgTlVMTCwgMCk7CiAgICAgICAgc3JjVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBzcmNsZW5XICogc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgaWYgKCFzcmNXKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzcmMsIHNyY2xlbiwgc3JjVywgc3JjbGVuVyk7CiAgICB9CgogICAgaWYgKGZsYWdzICYgTENNQVBfU09SVEtFWSkKICAgIHsKICAgICAgICBpZiAoc3JjID09IGRzdCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfZ2V0X3NvcnRrZXkoZmxhZ3MsIHNyY1csIHNyY2xlblcsIGRzdCwgZHN0bGVuKTsKICAgICAgICBnb3RvIG1hcF9zdHJpbmdfZXhpdDsKICAgIH0KCiAgICBpZiAoZmxhZ3MgJiBTT1JUX1NUUklOR1NPUlQpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIGdvdG8gbWFwX3N0cmluZ19leGl0OwogICAgfQoKICAgIGRzdGxlblcgPSBMQ01hcFN0cmluZ1cobGNpZCwgZmxhZ3MsIHNyY1csIHNyY2xlblcsIE5VTEwsIDApOwogICAgaWYgKCFkc3RsZW5XKQogICAgICAgIGdvdG8gbWFwX3N0cmluZ19leGl0OwoKICAgIGRzdFcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgZHN0bGVuVyAqIHNpemVvZihXQ0hBUikpOwogICAgaWYgKCFkc3RXKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CiAgICB9CgogICAgTENNYXBTdHJpbmdXKGxjaWQsIGZsYWdzLCBzcmNXLCBzcmNsZW5XLCBkc3RXLCBkc3RsZW5XKTsKICAgIHJldCA9IFdpZGVDaGFyVG9NdWx0aUJ5dGUobG9jYWxlX2NwLCAwLCBkc3RXLCBkc3RsZW5XLCBkc3QsIGRzdGxlbiwgTlVMTCwgTlVMTCk7CiAgICBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBkc3RXKTsKCm1hcF9zdHJpbmdfZXhpdDoKICAgIGlmIChzcmNXICE9IGJ1ZlcpIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEZvbGRTdHJpbmdBICAgIChLRVJORUwzMi5AKQogKgogKiBNYXAgY2hhcmFjdGVycyBpbiBhIHN0cmluZy4KICoKICogUEFSQU1TCiAqICBkd0ZsYWdzIFtJXSBGbGFncyBjb250cm9sbGluZyBjaGFycyB0byBtYXAgKE1BUF8gY29uc3RhbnRzIGZyb20gIndpbm5scy5oIikKICogIHNyYyAgICAgW0ldIFN0cmluZyB0byBtYXAKICogIHNyY2xlbiAgW0ldIExlbmd0aCBvZiBzcmMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgZHN0ICAgICBbT10gRGVzdGluYXRpb24gZm9yIG1hcHBlZCBzdHJpbmcKICogIGRzdGxlbiAgW0ldIExlbmd0aCBvZiBkc3QsIG9yIDAgdG8gZmluZCB0aGUgcmVxdWlyZWQgbGVuZ3RoIGZvciB0aGUgbWFwcGVkIHN0cmluZwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgbGVuZ3RoIG9mIHRoZSBzdHJpbmcgd3JpdHRlbiB0byBkc3QsIGluY2x1ZGluZyB0aGUgdGVybWluYXRpbmcgTlVMLiBJZgogKiAgICAgICAgICAgZHN0bGVuIGlzIDAsIHRoZSB2YWx1ZSByZXR1cm5lZCBpcyB0aGUgc2FtZSwgYnV0IG5vdGhpbmcgaXMgd3JpdHRlbiB0byBkc3QsCiAqICAgICAgICAgICBhbmQgZHN0IG1heSBiZSBOVUxMLgogKiAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpJTlQgV0lOQVBJIEZvbGRTdHJpbmdBKERXT1JEIGR3RmxhZ3MsIExQQ1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgTFBTVFIgZHN0LCBJTlQgZHN0bGVuKQp7CiAgICBJTlQgcmV0ID0gMCwgc3JjbGVuVyA9IDA7CiAgICBXQ0hBUiAqc3JjVyA9IE5VTEwsICpkc3RXID0gTlVMTDsKCiAgICBpZiAoIXNyYyB8fCAhc3JjbGVuIHx8IGRzdGxlbiA8IDAgfHwgKGRzdGxlbiAmJiAhZHN0KSB8fCBzcmMgPT0gZHN0KQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgc3JjbGVuVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIoQ1BfQUNQLCBkd0ZsYWdzICYgTUFQX0NPTVBPU0lURSA/IE1CX0NPTVBPU0lURSA6IDAsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzcmMsIHNyY2xlbiwgTlVMTCwgMCk7CiAgICBzcmNXID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY2xlblcgKiBzaXplb2YoV0NIQVIpKTsKCiAgICBpZiAoIXNyY1cpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICBnb3RvIEZvbGRTdHJpbmdBX2V4aXQ7CiAgICB9CgogICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIGR3RmxhZ3MgJiBNQVBfQ09NUE9TSVRFID8gTUJfQ09NUE9TSVRFIDogMCwKICAgICAgICAgICAgICAgICAgICAgICAgc3JjLCBzcmNsZW4sIHNyY1csIHNyY2xlblcpOwoKICAgIGR3RmxhZ3MgPSAoZHdGbGFncyAmIH5NQVBfUFJFQ09NUE9TRUQpIHwgTUFQX0ZPTERDWk9ORTsKCiAgICByZXQgPSBGb2xkU3RyaW5nVyhkd0ZsYWdzLCBzcmNXLCBzcmNsZW5XLCBOVUxMLCAwKTsKICAgIGlmIChyZXQgJiYgZHN0bGVuKQogICAgewogICAgICAgIGRzdFcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgcmV0ICogc2l6ZW9mKFdDSEFSKSk7CgogICAgICAgIGlmICghZHN0VykKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgICAgIGdvdG8gRm9sZFN0cmluZ0FfZXhpdDsKICAgICAgICB9CgogICAgICAgIHJldCA9IEZvbGRTdHJpbmdXKGR3RmxhZ3MsIHNyY1csIHNyY2xlblcsIGRzdFcsIHJldCk7CiAgICAgICAgaWYgKCFXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgZHN0VywgcmV0LCBkc3QsIGRzdGxlbiwgTlVMTCwgTlVMTCkpCiAgICAgICAgewogICAgICAgICAgICByZXQgPSAwOwogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUik7CiAgICAgICAgfQogICAgfQoKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGRzdFcpOwoKRm9sZFN0cmluZ0FfZXhpdDoKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEZvbGRTdHJpbmdXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRm9sZFN0cmluZ0EuCiAqLwpJTlQgV0lOQVBJIEZvbGRTdHJpbmdXKERXT1JEIGR3RmxhZ3MsIExQQ1dTVFIgc3JjLCBJTlQgc3JjbGVuLAogICAgICAgICAgICAgICAgICAgICAgIExQV1NUUiBkc3QsIElOVCBkc3RsZW4pCnsKICAgIGludCByZXQ7CgogICAgc3dpdGNoIChkd0ZsYWdzICYgKE1BUF9DT01QT1NJVEV8TUFQX1BSRUNPTVBPU0VEfE1BUF9FWFBBTkRfTElHQVRVUkVTKSkKICAgIHsKICAgIGNhc2UgMDoKICAgICAgICBpZiAoZHdGbGFncykKICAgICAgICAgIGJyZWFrOwogICAgICAgIC8qIEZhbGwgdGhyb3VnaCBmb3IgZHdGbGFncyA9PSAwICovCiAgICBjYXNlIE1BUF9QUkVDT01QT1NFRHxNQVBfQ09NUE9TSVRFOgogICAgY2FzZSBNQVBfUFJFQ09NUE9TRUR8TUFQX0VYUEFORF9MSUdBVFVSRVM6CiAgICBjYXNlIE1BUF9DT01QT1NJVEV8TUFQX0VYUEFORF9MSUdBVFVSRVM6CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmICghc3JjIHx8ICFzcmNsZW4gfHwgZHN0bGVuIDwgMCB8fCAoZHN0bGVuICYmICFkc3QpIHx8IHNyYyA9PSBkc3QpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICByZXQgPSB3aW5lX2ZvbGRfc3RyaW5nKGR3RmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbik7CiAgICBpZiAoIXJldCkKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUik7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBDb21wYXJlU3RyaW5nVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIENvbXBhcmVTdHJpbmdBLgogKi8KSU5UIFdJTkFQSSBDb21wYXJlU3RyaW5nVyhMQ0lEIGxjaWQsIERXT1JEIHN0eWxlLAogICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1dTVFIgc3RyMSwgSU5UIGxlbjEsIExQQ1dTVFIgc3RyMiwgSU5UIGxlbjIpCnsKICAgIElOVCByZXQ7CgogICAgaWYgKCFzdHIxIHx8ICFzdHIyKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYoIHN0eWxlICYgfihOT1JNX0lHTk9SRUNBU0V8Tk9STV9JR05PUkVOT05TUEFDRXxOT1JNX0lHTk9SRVNZTUJPTFN8CiAgICAgICAgU09SVF9TVFJJTkdTT1JUfE5PUk1fSUdOT1JFS0FOQVRZUEV8Tk9STV9JR05PUkVXSURUSHxMT0NBTEVfVVNFX0NQX0FDUHwweDEwMDAwMDAwKSApCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzdHlsZSAmIDB4MTAwMDAwMDApCiAgICAgICAgRklYTUUoIklnbm9yaW5nIHVua25vd24gc3R5bGUgMHgxMDAwMDAwMFxuIik7CgogICAgaWYgKGxlbjEgPCAwKSBsZW4xID0gc3RybGVuVyhzdHIxKTsKICAgIGlmIChsZW4yIDwgMCkgbGVuMiA9IHN0cmxlblcoc3RyMik7CgogICAgcmV0ID0gd2luZV9jb21wYXJlX3N0cmluZyhzdHlsZSwgc3RyMSwgbGVuMSwgc3RyMiwgbGVuMik7CgogICAgaWYgKHJldCkgLyogbmVlZCB0byB0cmFuc2xhdGUgcmVzdWx0ICovCiAgICAgICAgcmV0dXJuIChyZXQgPCAwKSA/IENTVFJfTEVTU19USEFOIDogQ1NUUl9HUkVBVEVSX1RIQU47CiAgICByZXR1cm4gQ1NUUl9FUVVBTDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgQ29tcGFyZVN0cmluZ0EgICAgKEtFUk5FTDMyLkApCiAqCiAqIENvbXBhcmUgdHdvIGxvY2FsZSBzZW5zaXRpdmUgc3RyaW5ncy4KICoKICogUEFSQU1TCiAqICBsY2lkICBbSV0gTENJRCBmb3IgdGhlIGNvbXBhcmlzb24KICogIHN0eWxlIFtJXSBGbGFncyBmb3IgdGhlIGNvbXBhcmlzb24gKE5PUk1fIGNvbnN0YW50cyBmcm9tICJ3aW5ubHMuaCIpLgogKiAgc3RyMSAgW0ldIEZpcnN0IHN0cmluZyB0byBjb21wYXJlCiAqICBsZW4xICBbSV0gTGVuZ3RoIG9mIHN0cjEsIG9yIC0xIGlmIHN0cjEgaXMgTlVMIHRlcm1pbmF0ZWQKICogIHN0cjIgIFtJXSBTZWNvbmQgc3RyaW5nIHRvIGNvbXBhcmUKICogIGxlbjIgIFtJXSBMZW5ndGggb2Ygc3RyMiwgb3IgLTEgaWYgc3RyMiBpcyBOVUwgdGVybWluYXRlZAogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBDU1RSX0xFU1NfVEhBTiwgQ1NUUl9FUVVBTCBvciBDU1RSX0dSRUFURVJfVEhBTiBkZXBlbmRpbmcgb24gd2hldGhlcgogKiAgICAgICAgICAgc3RyMiBpcyBsZXNzIHRoYW4sIGVxdWFsIHRvIG9yIGdyZWF0ZXIgdGhhbiBzdHIxIHJlc3BlY3RpdmVseS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCklOVCBXSU5BUEkgQ29tcGFyZVN0cmluZ0EoTENJRCBsY2lkLCBEV09SRCBzdHlsZSwKICAgICAgICAgICAgICAgICAgICAgICAgICBMUENTVFIgc3RyMSwgSU5UIGxlbjEsIExQQ1NUUiBzdHIyLCBJTlQgbGVuMikKewogICAgV0NIQVIgKmJ1ZjFXID0gTnRDdXJyZW50VGViKCktPlN0YXRpY1VuaWNvZGVCdWZmZXI7CiAgICBXQ0hBUiAqYnVmMlcgPSBidWYxVyArIDEzMDsKICAgIExQV1NUUiBzdHIxVywgc3RyMlc7CiAgICBJTlQgbGVuMVcsIGxlbjJXLCByZXQ7CiAgICBVSU5UIGxvY2FsZV9jcCA9IENQX0FDUDsKCiAgICBpZiAoIXN0cjEgfHwgIXN0cjIpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIGlmIChsZW4xIDwgMCkgbGVuMSA9IHN0cmxlbihzdHIxKTsKICAgIGlmIChsZW4yIDwgMCkgbGVuMiA9IHN0cmxlbihzdHIyKTsKCiAgICBpZiAoIShzdHlsZSAmIExPQ0FMRV9VU0VfQ1BfQUNQKSkgbG9jYWxlX2NwID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKCiAgICBsZW4xVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIxLCBsZW4xLCBidWYxVywgMTMwKTsKICAgIGlmIChsZW4xVykKICAgICAgICBzdHIxVyA9IGJ1ZjFXOwogICAgZWxzZQogICAgewogICAgICAgIGxlbjFXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjEsIGxlbjEsIE5VTEwsIDApOwogICAgICAgIHN0cjFXID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIGxlbjFXICogc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgaWYgKCFzdHIxVykKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KICAgICAgICBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMSwgbGVuMSwgc3RyMVcsIGxlbjFXKTsKICAgIH0KICAgIGxlbjJXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjIsIGxlbjIsIGJ1ZjJXLCAxMzApOwogICAgaWYgKGxlbjJXKQogICAgICAgIHN0cjJXID0gYnVmMlc7CiAgICBlbHNlCiAgICB7CiAgICAgICAgbGVuMlcgPSBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMiwgbGVuMiwgTlVMTCwgMCk7CiAgICAgICAgc3RyMlcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGVuMlcgKiBzaXplb2YoV0NIQVIpKTsKICAgICAgICBpZiAoIXN0cjJXKQogICAgICAgIHsKICAgICAgICAgICAgaWYgKHN0cjFXICE9IGJ1ZjFXKSBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHIxVyk7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KICAgICAgICBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMiwgbGVuMiwgc3RyMlcsIGxlbjJXKTsKICAgIH0KCiAgICByZXQgPSBDb21wYXJlU3RyaW5nVyhsY2lkLCBzdHlsZSwgc3RyMVcsIGxlbjFXLCBzdHIyVywgbGVuMlcpOwoKICAgIGlmIChzdHIxVyAhPSBidWYxVykgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3RyMVcpOwogICAgaWYgKHN0cjJXICE9IGJ1ZjJXKSBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHIyVyk7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgbHN0cmNtcCAgICAgKEtFUk5FTDMyLkApCiAqICAgICAgICAgICBsc3RyY21wQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ29tcGFyZSB0d28gc3RyaW5ncyB1c2luZyB0aGUgY3VycmVudCB0aHJlYWQgbG9jYWxlLgogKgogKiBQQVJBTVMKICogIHN0cjEgIFtJXSBGaXJzdCBzdHJpbmcgdG8gY29tcGFyZQogKiAgc3RyMiAgW0ldIFNlY29uZCBzdHJpbmcgdG8gY29tcGFyZQogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBBIG51bWJlciBsZXNzIHRoYW4sIGVxdWFsIHRvIG9yIGdyZWF0ZXIgdGhhbiAwIGRlcGVuZGluZyBvbiB3aGV0aGVyCiAqICAgICAgICAgICBzdHIyIGlzIGxlc3MgdGhhbiwgZXF1YWwgdG8gb3IgZ3JlYXRlciB0aGFuIHN0cjEgcmVzcGVjdGl2ZWx5LgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KaW50IFdJTkFQSSBsc3RyY21wQShMUENTVFIgc3RyMSwgTFBDU1RSIHN0cjIpCnsKICAgIGludCByZXQ7CiAgICAKICAgIGlmICgoc3RyMSA9PSBOVUxMKSAmJiAoc3RyMiA9PSBOVUxMKSkgcmV0dXJuIDA7CiAgICBpZiAoc3RyMSA9PSBOVUxMKSByZXR1cm4gLTE7CiAgICBpZiAoc3RyMiA9PSBOVUxMKSByZXR1cm4gMTsKCiAgICByZXQgPSBDb21wYXJlU3RyaW5nQShHZXRUaHJlYWRMb2NhbGUoKSwgTE9DQUxFX1VTRV9DUF9BQ1AsIHN0cjEsIC0xLCBzdHIyLCAtMSk7CiAgICBpZiAocmV0KSByZXQgLT0gMjsKICAgIAogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGxzdHJjbXBpICAgICAoS0VSTkVMMzIuQCkKICogICAgICAgICAgIGxzdHJjbXBpQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ29tcGFyZSB0d28gc3RyaW5ncyB1c2luZyB0aGUgY3VycmVudCB0aHJlYWQgbG9jYWxlLCBpZ25vcmluZyBjYXNlLgogKgogKiBQQVJBTVMKICogIHN0cjEgIFtJXSBGaXJzdCBzdHJpbmcgdG8gY29tcGFyZQogKiAgc3RyMiAgW0ldIFNlY29uZCBzdHJpbmcgdG8gY29tcGFyZQogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBBIG51bWJlciBsZXNzIHRoYW4sIGVxdWFsIHRvIG9yIGdyZWF0ZXIgdGhhbiAwIGRlcGVuZGluZyBvbiB3aGV0aGVyCiAqICAgICAgICAgICBzdHIyIGlzIGxlc3MgdGhhbiwgZXF1YWwgdG8gb3IgZ3JlYXRlciB0aGFuIHN0cjEgcmVzcGVjdGl2ZWx5LgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KaW50IFdJTkFQSSBsc3RyY21waUEoTFBDU1RSIHN0cjEsIExQQ1NUUiBzdHIyKQp7CiAgICBpbnQgcmV0OwogICAgCiAgICBpZiAoKHN0cjEgPT0gTlVMTCkgJiYgKHN0cjIgPT0gTlVMTCkpIHJldHVybiAwOwogICAgaWYgKHN0cjEgPT0gTlVMTCkgcmV0dXJuIC0xOwogICAgaWYgKHN0cjIgPT0gTlVMTCkgcmV0dXJuIDE7CgogICAgcmV0ID0gQ29tcGFyZVN0cmluZ0EoR2V0VGhyZWFkTG9jYWxlKCksIE5PUk1fSUdOT1JFQ0FTRXxMT0NBTEVfVVNFX0NQX0FDUCwgc3RyMSwgLTEsIHN0cjIsIC0xKTsKICAgIGlmIChyZXQpIHJldCAtPSAyOwogICAgCiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgbHN0cmNtcFcgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBsc3RyY21wQS4KICovCmludCBXSU5BUEkgbHN0cmNtcFcoTFBDV1NUUiBzdHIxLCBMUENXU1RSIHN0cjIpCnsKICAgIGludCByZXQ7CgogICAgaWYgKChzdHIxID09IE5VTEwpICYmIChzdHIyID09IE5VTEwpKSByZXR1cm4gMDsKICAgIGlmIChzdHIxID09IE5VTEwpIHJldHVybiAtMTsKICAgIGlmIChzdHIyID09IE5VTEwpIHJldHVybiAxOwoKICAgIHJldCA9IENvbXBhcmVTdHJpbmdXKEdldFRocmVhZExvY2FsZSgpLCAwLCBzdHIxLCAtMSwgc3RyMiwgLTEpOwogICAgaWYgKHJldCkgcmV0IC09IDI7CiAgICAKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBsc3RyY21waVcgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBsc3RyY21waUEuCiAqLwppbnQgV0lOQVBJIGxzdHJjbXBpVyhMUENXU1RSIHN0cjEsIExQQ1dTVFIgc3RyMikKewogICAgaW50IHJldDsKICAgIAogICAgaWYgKChzdHIxID09IE5VTEwpICYmIChzdHIyID09IE5VTEwpKSByZXR1cm4gMDsKICAgIGlmIChzdHIxID09IE5VTEwpIHJldHVybiAtMTsKICAgIGlmIChzdHIyID09IE5VTEwpIHJldHVybiAxOwoKICAgIHJldCA9IENvbXBhcmVTdHJpbmdXKEdldFRocmVhZExvY2FsZSgpLCBOT1JNX0lHTk9SRUNBU0UsIHN0cjEsIC0xLCBzdHIyLCAtMSk7CiAgICBpZiAocmV0KSByZXQgLT0gMjsKICAgIAogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJTE9DQUxFX0luaXQKICovCnZvaWQgTE9DQUxFX0luaXQodm9pZCkKewogICAgZXh0ZXJuIHZvaWQgX193aW5lX2luaXRfY29kZXBhZ2VzKCBjb25zdCB1bmlvbiBjcHRhYmxlICphbnNpX2NwLCBjb25zdCB1bmlvbiBjcHRhYmxlICpvZW1fY3AsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnVuaXhfY3AgKTsKCiAgICBVSU5UIGFuc2lfY3AgPSAxMjUyLCBvZW1fY3AgPSA0MzcsIG1hY19jcCA9IDEwMDAwLCB1bml4X2NwID0gfjBVOwogICAgTENJRCBsY2lkOwoKICAgIGxjaWQgPSBnZXRfZW52X2xjaWQoIE5VTEwsIE5VTEwgKTsKICAgIE50U2V0RGVmYXVsdExvY2FsZSggVFJVRSwgbGNpZCApOwoKICAgIGxjaWQgPSBnZXRfZW52X2xjaWQoIE5VTEwsICJMQ19NRVNTQUdFUyIgKTsKICAgIE50U2V0RGVmYXVsdFVJTGFuZ3VhZ2UoIExBTkdJREZST01MQ0lEKGxjaWQpICk7CgogICAgbGNpZCA9IGdldF9lbnZfbGNpZCggJnVuaXhfY3AsICJMQ19DVFlQRSIgKTsKICAgIE50U2V0RGVmYXVsdExvY2FsZSggRkFMU0UsIGxjaWQgKTsKCiAgICBhbnNpX2NwID0gZ2V0X2xjaWRfY29kZXBhZ2UobGNpZCk7CiAgICBHZXRMb2NhbGVJbmZvVyggbGNpZCwgTE9DQUxFX0lERUZBVUxUTUFDQ09ERVBBR0UgfCBMT0NBTEVfUkVUVVJOX05VTUJFUiwKICAgICAgICAgICAgICAgICAgICAoTFBXU1RSKSZtYWNfY3AsIHNpemVvZihtYWNfY3ApL3NpemVvZihXQ0hBUikgKTsKICAgIEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfSURFRkFVTFRDT0RFUEFHRSB8IExPQ0FMRV9SRVRVUk5fTlVNQkVSLAogICAgICAgICAgICAgICAgICAgIChMUFdTVFIpJm9lbV9jcCwgc2l6ZW9mKG9lbV9jcCkvc2l6ZW9mKFdDSEFSKSApOwogICAgaWYgKHVuaXhfY3AgPT0gfjBVKQogICAgICAgIEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfSURFRkFVTFRVTklYQ09ERVBBR0UgfCBMT0NBTEVfUkVUVVJOX05VTUJFUiwKICAgICAgICAgICAgICAgICAgICAoTFBXU1RSKSZ1bml4X2NwLCBzaXplb2YodW5peF9jcCkvc2l6ZW9mKFdDSEFSKSApOwoKICAgIGlmICghKGFuc2lfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBhbnNpX2NwICkpKQogICAgICAgIGFuc2lfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAxMjUyICk7CiAgICBpZiAoIShvZW1fY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBvZW1fY3AgKSkpCiAgICAgICAgb2VtX2NwdGFibGUgID0gd2luZV9jcF9nZXRfdGFibGUoIDQzNyApOwogICAgaWYgKCEobWFjX2NwdGFibGUgPSB3aW5lX2NwX2dldF90YWJsZSggbWFjX2NwICkpKQogICAgICAgIG1hY19jcHRhYmxlICA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAxMDAwMCApOwogICAgaWYgKHVuaXhfY3AgIT0gQ1BfVVRGOCkKICAgIHsKICAgICAgICBpZiAoISh1bml4X2NwdGFibGUgPSB3aW5lX2NwX2dldF90YWJsZSggdW5peF9jcCApKSkKICAgICAgICAgICAgdW5peF9jcHRhYmxlICA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAyODU5MSApOwogICAgfQoKICAgIF9fd2luZV9pbml0X2NvZGVwYWdlcyggYW5zaV9jcHRhYmxlLCBvZW1fY3B0YWJsZSwgdW5peF9jcHRhYmxlICk7CgogICAgVFJBQ0UoICJhbnNpPSUwM2Qgb2VtPSUwM2QgbWFjPSUwM2QgdW5peD0lMDNkXG4iLAogICAgICAgICAgIGFuc2lfY3B0YWJsZS0+aW5mby5jb2RlcGFnZSwgb2VtX2NwdGFibGUtPmluZm8uY29kZXBhZ2UsCiAgICAgICAgICAgbWFjX2NwdGFibGUtPmluZm8uY29kZXBhZ2UsIHVuaXhfY3AgKTsKfQoKc3RhdGljIEhBTkRMRSBOTFNfUmVnT3BlbktleShIQU5ETEUgaFJvb3RLZXksIExQQ1dTVFIgc3pLZXlOYW1lKQp7CiAgICBVTklDT0RFX1NUUklORyBrZXlOYW1lOwogICAgT0JKRUNUX0FUVFJJQlVURVMgYXR0cjsKICAgIEhBTkRMRSBoa2V5OwoKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAma2V5TmFtZSwgc3pLZXlOYW1lICk7CiAgICBJbml0aWFsaXplT2JqZWN0QXR0cmlidXRlcygmYXR0ciwgJmtleU5hbWUsIDAsIGhSb290S2V5LCBOVUxMKTsKCiAgICBpZiAoTnRPcGVuS2V5KCAmaGtleSwgS0VZX0FMTF9BQ0NFU1MsICZhdHRyICkgIT0gU1RBVFVTX1NVQ0NFU1MpCiAgICAgICAgaGtleSA9IDA7CgogICAgcmV0dXJuIGhrZXk7Cn0KCnN0YXRpYyBIQU5ETEUgTkxTX1JlZ09wZW5TdWJLZXkoSEFORExFIGhSb290S2V5LCBMUENXU1RSIHN6S2V5TmFtZSkKewogICAgSEFORExFIGhLZXkgPSBOTFNfUmVnT3BlbktleShoUm9vdEtleSwgc3pLZXlOYW1lKTsKCiAgICBpZiAoaFJvb3RLZXkpCiAgICAgICAgTnRDbG9zZSggaFJvb3RLZXkgKTsKCiAgICByZXR1cm4gaEtleTsKfQoKc3RhdGljIEJPT0wgTkxTX1JlZ0VudW1TdWJLZXkoSEFORExFIGhLZXksIFVJTlQgdWxJbmRleCwgTFBXU1RSIHN6S2V5TmFtZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUxPTkcga2V5TmFtZVNpemUpCnsKICAgIEJZVEUgYnVmZmVyWzgwXTsKICAgIEtFWV9CQVNJQ19JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfQkFTSUNfSU5GT1JNQVRJT04gKilidWZmZXI7CiAgICBEV09SRCBkd0xlbjsKCiAgICBpZiAoTnRFbnVtZXJhdGVLZXkoIGhLZXksIHVsSW5kZXgsIEtleUJhc2ljSW5mb3JtYXRpb24sIGJ1ZmZlciwKICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGJ1ZmZlciksICZkd0xlbikgIT0gU1RBVFVTX1NVQ0NFU1MgfHwKICAgICAgICBpbmZvLT5OYW1lTGVuZ3RoID4ga2V5TmFtZVNpemUpCiAgICB7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIFRSQUNFKCJpbmZvLT5OYW1lICVzIGluZm8tPk5hbWVMZW5ndGggJWxkXG4iLCBkZWJ1Z3N0cl93KGluZm8tPk5hbWUpLCBpbmZvLT5OYW1lTGVuZ3RoKTsKCiAgICBtZW1jcHkoIHN6S2V5TmFtZSwgaW5mby0+TmFtZSwgaW5mby0+TmFtZUxlbmd0aCk7CiAgICBzektleU5hbWVbaW5mby0+TmFtZUxlbmd0aCAvIHNpemVvZihXQ0hBUildID0gJ1wwJzsKCiAgICBUUkFDRSgicmV0dXJuaW5nICVzXG4iLCBkZWJ1Z3N0cl93KHN6S2V5TmFtZSkpOwogICAgcmV0dXJuIFRSVUU7Cn0KCnN0YXRpYyBCT09MIE5MU19SZWdFbnVtVmFsdWUoSEFORExFIGhLZXksIFVJTlQgdWxJbmRleCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUFdTVFIgc3pWYWx1ZU5hbWUsIFVMT05HIHZhbHVlTmFtZVNpemUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBXU1RSIHN6VmFsdWVEYXRhLCBVTE9ORyB2YWx1ZURhdGFTaXplKQp7CiAgICBCWVRFIGJ1ZmZlcls4MF07CiAgICBLRVlfVkFMVUVfRlVMTF9JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfVkFMVUVfRlVMTF9JTkZPUk1BVElPTiAqKWJ1ZmZlcjsKICAgIERXT1JEIGR3TGVuOwoKICAgIGlmIChOdEVudW1lcmF0ZVZhbHVlS2V5KCBoS2V5LCB1bEluZGV4LCBLZXlWYWx1ZUZ1bGxJbmZvcm1hdGlvbiwKICAgICAgICBidWZmZXIsIHNpemVvZihidWZmZXIpLCAmZHdMZW4gKSAhPSBTVEFUVVNfU1VDQ0VTUyB8fAogICAgICAgIGluZm8tPk5hbWVMZW5ndGggPiB2YWx1ZU5hbWVTaXplIHx8CiAgICAgICAgaW5mby0+RGF0YUxlbmd0aCA+IHZhbHVlRGF0YVNpemUpCiAgICB7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIFRSQUNFKCJpbmZvLT5OYW1lICVzIGluZm8tPkRhdGFMZW5ndGggJWxkXG4iLCBkZWJ1Z3N0cl93KGluZm8tPk5hbWUpLCBpbmZvLT5EYXRhTGVuZ3RoKTsKCiAgICBtZW1jcHkoIHN6VmFsdWVOYW1lLCBpbmZvLT5OYW1lLCBpbmZvLT5OYW1lTGVuZ3RoKTsKICAgIHN6VmFsdWVOYW1lW2luZm8tPk5hbWVMZW5ndGggLyBzaXplb2YoV0NIQVIpXSA9ICdcMCc7CiAgICBtZW1jcHkoIHN6VmFsdWVEYXRhLCBidWZmZXIgKyBpbmZvLT5EYXRhT2Zmc2V0LCBpbmZvLT5EYXRhTGVuZ3RoICk7CiAgICBzelZhbHVlRGF0YVtpbmZvLT5EYXRhTGVuZ3RoIC8gc2l6ZW9mKFdDSEFSKV0gPSAnXDAnOwoKICAgIFRSQUNFKCJyZXR1cm5pbmcgJXMgJXNcbiIsIGRlYnVnc3RyX3coc3pWYWx1ZU5hbWUpLCBkZWJ1Z3N0cl93KHN6VmFsdWVEYXRhKSk7CiAgICByZXR1cm4gVFJVRTsKfQoKc3RhdGljIEJPT0wgTkxTX1JlZ0dldER3b3JkKEhBTkRMRSBoS2V5LCBMUENXU1RSIHN6VmFsdWVOYW1lLCBEV09SRCAqbHBWYWwpCnsKICAgIEJZVEUgYnVmZmVyWzEyOF07CiAgICBjb25zdCBLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqKWJ1ZmZlcjsKICAgIERXT1JEIGR3U2l6ZSA9IHNpemVvZihidWZmZXIpOwogICAgVU5JQ09ERV9TVFJJTkcgdmFsdWVOYW1lOwoKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmdmFsdWVOYW1lLCBzelZhbHVlTmFtZSApOwoKICAgIFRSQUNFKCIlcCwgJXNcbiIsIGhLZXksIGRlYnVnc3RyX3coc3pWYWx1ZU5hbWUpKTsKICAgIGlmIChOdFF1ZXJ5VmFsdWVLZXkoIGhLZXksICZ2YWx1ZU5hbWUsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLAogICAgICAgICAgICAgICAgICAgICAgICAgYnVmZmVyLCBkd1NpemUsICZkd1NpemUgKSA9PSBTVEFUVVNfU1VDQ0VTUyAmJgogICAgICAgIGluZm8tPkRhdGFMZW5ndGggPT0gc2l6ZW9mKERXT1JEKSkKICAgIHsKICAgICAgICBtZW1jcHkobHBWYWwsIGluZm8tPkRhdGEsIHNpemVvZihEV09SRCkpOwogICAgICAgIHJldHVybiBUUlVFOwogICAgfQoKICAgIHJldHVybiBGQUxTRTsKfQoKc3RhdGljIEJPT0wgTkxTX0dldExhbmd1YWdlR3JvdXBOYW1lKExHUlBJRCBsZ3JwaWQsIExQV1NUUiBzek5hbWUsIFVMT05HIG5hbWVTaXplKQp7CiAgICBMQU5HSUQgIGxhbmdJZDsKICAgIExQQ1dTVFIgc3pSZXNvdXJjZU5hbWUgPSBNQUtFSU5UUkVTT1VSQ0VXKCgobGdycGlkICsgMHgyMDAwKSA+PiA0KSArIDEpOwogICAgSFJTUkMgICBoUmVzb3VyY2U7CiAgICBCT09MICAgIGJSZXQgPSBGQUxTRTsKCiAgICAvKiBGSVhNRTogSXMgaXQgY29ycmVjdCB0byB1c2UgdGhlIHN5c3RlbSBkZWZhdWx0IGxhbmdpZD8gKi8KICAgIGxhbmdJZCA9IEdldFN5c3RlbURlZmF1bHRMYW5nSUQoKTsKCiAgICBpZiAoU1VCTEFOR0lEKGxhbmdJZCkgPT0gU1VCTEFOR19ORVVUUkFMKQogICAgICAgIGxhbmdJZCA9IE1BS0VMQU5HSUQoIFBSSU1BUllMQU5HSUQobGFuZ0lkKSwgU1VCTEFOR19ERUZBVUxUICk7CgogICAgaFJlc291cmNlID0gRmluZFJlc291cmNlRXhXKCBrZXJuZWwzMl9oYW5kbGUsIChMUFdTVFIpUlRfU1RSSU5HLCBzelJlc291cmNlTmFtZSwgbGFuZ0lkICk7CgogICAgaWYgKGhSZXNvdXJjZSkKICAgIHsKICAgICAgICBIR0xPQkFMIGhSZXNEaXIgPSBMb2FkUmVzb3VyY2UoIGtlcm5lbDMyX2hhbmRsZSwgaFJlc291cmNlICk7CgogICAgICAgIGlmIChoUmVzRGlyKQogICAgICAgIHsKICAgICAgICAgICAgVUxPTkcgICBpUmVzb3VyY2VJbmRleCA9IGxncnBpZCAmIDB4ZjsKICAgICAgICAgICAgTFBDV1NUUiBscFJlc0VudHJ5ID0gTG9ja1Jlc291cmNlKCBoUmVzRGlyICk7CiAgICAgICAgICAgIFVMT05HICAgaTsKCiAgICAgICAgICAgIGZvciAoaSA9IDA7IGkgPCBpUmVzb3VyY2VJbmRleDsgaSsrKQogICAgICAgICAgICAgICAgbHBSZXNFbnRyeSArPSAqbHBSZXNFbnRyeSArIDE7CgogICAgICAgICAgICBpZiAoKmxwUmVzRW50cnkgPCBuYW1lU2l6ZSkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgbWVtY3B5KCBzek5hbWUsIGxwUmVzRW50cnkgKyAxLCAqbHBSZXNFbnRyeSAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICAgICAgICAgIHN6TmFtZVsqbHBSZXNFbnRyeV0gPSAnXDAnOwogICAgICAgICAgICAgICAgYlJldCA9IFRSVUU7CiAgICAgICAgICAgIH0KCiAgICAgICAgfQogICAgICAgIEZyZWVSZXNvdXJjZSggaFJlc291cmNlICk7CiAgICB9CiAgICByZXR1cm4gYlJldDsKfQoKLyogUmVnaXN0cnkga2V5cyBmb3IgTkxTIHJlbGF0ZWQgaW5mb3JtYXRpb24gKi8Kc3RhdGljIGNvbnN0IFdDSEFSIHN6TGFuZ0dyb3Vwc0tleU5hbWVbXSA9IHsKICAgICdMJywnYScsJ24nLCdnJywndScsJ2EnLCdnJywnZScsJyAnLCdHJywncicsJ28nLCd1JywncCcsJ3MnLCdcMCcKfTsKCnN0YXRpYyBjb25zdCBXQ0hBUiBzekNvdW50cnlMaXN0TmFtZVtdID0gewogICAgJ00nLCdhJywnYycsJ2gnLCdpJywnbicsJ2UnLCdcXCcsJ1MnLCdvJywnZicsJ3QnLCd3JywnYScsJ3InLCdlJywnXFwnLAogICAgJ00nLCdpJywnYycsJ3InLCdvJywncycsJ28nLCdmJywndCcsJ1xcJywnVycsJ2knLCduJywnZCcsJ28nLCd3JywncycsJ1xcJywKICAgICdDJywndScsJ3InLCdyJywnZScsJ24nLCd0JywnVicsJ2UnLCdyJywncycsJ2knLCdvJywnbicsJ1xcJywKICAgICdUJywnZScsJ2wnLCdlJywncCcsJ2gnLCdvJywnbicsJ3knLCdcXCcsCiAgICAnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsJyAnLCdMJywnaScsJ3MnLCd0JywnXDAnCn07CgoKLyogQ2FsbGJhY2sgZnVuY3Rpb24gcHRycyBmb3IgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzQS9XICovCnR5cGVkZWYgc3RydWN0CnsKICBMQU5HVUFHRUdST1VQX0VOVU1QUk9DQSBwcm9jQTsKICBMQU5HVUFHRUdST1VQX0VOVU1QUk9DVyBwcm9jVzsKICBEV09SRCAgICBkd0ZsYWdzOwogIExPTkdfUFRSIGxQYXJhbTsKfSBFTlVNTEFOR1VBR0VHUk9VUF9DQUxMQkFDS1M7CgovKiBJbnRlcm5hbCBpbXBsZW1lbnRhdGlvbiBvZiBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBL1cgKi8Kc3RhdGljIEJPT0wgTkxTX0VudW1TeXN0ZW1MYW5ndWFnZUdyb3VwcyhFTlVNTEFOR1VBR0VHUk9VUF9DQUxMQkFDS1MgKmxwUHJvY3MpCnsKICAgIFdDSEFSIHN6TnVtYmVyWzEwXSwgc3pWYWx1ZVs0XTsKICAgIEhBTkRMRSBoS2V5OwogICAgQk9PTCBiQ29udGludWUgPSBUUlVFOwogICAgVUxPTkcgdWxJbmRleCA9IDA7CgogICAgaWYgKCFscFByb2NzKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIHN3aXRjaCAobHBQcm9jcy0+ZHdGbGFncykKICAgIHsKICAgIGNhc2UgMDoKICAgICAgICAvKiBEZWZhdWx0IHRvIExHUlBJRF9JTlNUQUxMRUQgKi8KICAgICAgICBscFByb2NzLT5kd0ZsYWdzID0gTEdSUElEX0lOU1RBTExFRDsKICAgICAgICAvKiBGYWxsIHRocm91Z2guLi4gKi8KICAgIGNhc2UgTEdSUElEX0lOU1RBTExFRDoKICAgIGNhc2UgTEdSUElEX1NVUFBPUlRFRDoKICAgICAgICBicmVhazsKICAgIGRlZmF1bHQ6CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBoS2V5ID0gTkxTX1JlZ09wZW5TdWJLZXkoIE5MU19SZWdPcGVuS2V5KCAwLCBzek5sc0tleU5hbWUgKSwgc3pMYW5nR3JvdXBzS2V5TmFtZSApOwoKICAgIGlmICghaEtleSkKICAgICAgICBGSVhNRSgiTkxTIHJlZ2lzdHJ5IGtleSBub3QgZm91bmQuIFBsZWFzZSBhcHBseSB0aGUgZGVmYXVsdCByZWdpc3RyeSBmaWxlICd3aW5lLmluZidcbiIpOwoKICAgIHdoaWxlIChiQ29udGludWUpCiAgICB7CiAgICAgICAgaWYgKE5MU19SZWdFbnVtVmFsdWUoIGhLZXksIHVsSW5kZXgsIHN6TnVtYmVyLCBzaXplb2Yoc3pOdW1iZXIpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzelZhbHVlLCBzaXplb2Yoc3pWYWx1ZSkgKSkKICAgICAgICB7CiAgICAgICAgICAgIEJPT0wgYkluc3RhbGxlZCA9IHN6VmFsdWVbMF0gPT0gJzEnID8gVFJVRSA6IEZBTFNFOwogICAgICAgICAgICBMR1JQSUQgbGdycGlkID0gc3RydG91bFcoIHN6TnVtYmVyLCBOVUxMLCAxNiApOwoKICAgICAgICAgICAgVFJBQ0UoImdycGlkICVzICglc2luc3RhbGxlZClcbiIsIGRlYnVnc3RyX3coc3pOdW1iZXIpLAogICAgICAgICAgICAgICAgICAgYkluc3RhbGxlZCA/ICIiIDogIm5vdCAiKTsKCiAgICAgICAgICAgIGlmIChscFByb2NzLT5kd0ZsYWdzID09IExHUlBJRF9TVVBQT1JURUQgfHwgYkluc3RhbGxlZCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NIQVIgc3pHcnBOYW1lWzQ4XTsKCiAgICAgICAgICAgICAgICBpZiAoIU5MU19HZXRMYW5ndWFnZUdyb3VwTmFtZSggbGdycGlkLCBzekdycE5hbWUsIHNpemVvZihzekdycE5hbWUpIC8gc2l6ZW9mKFdDSEFSKSApKQogICAgICAgICAgICAgICAgICAgIHN6R3JwTmFtZVswXSA9ICdcMCc7CgogICAgICAgICAgICAgICAgaWYgKGxwUHJvY3MtPnByb2NXKQogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NXKCBsZ3JwaWQsIHN6TnVtYmVyLCBzekdycE5hbWUsIGxwUHJvY3MtPmR3RmxhZ3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxwUHJvY3MtPmxQYXJhbSApOwogICAgICAgICAgICAgICAgZWxzZQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgIGNoYXIgc3pOdW1iZXJBW3NpemVvZihzek51bWJlcikvc2l6ZW9mKFdDSEFSKV07CiAgICAgICAgICAgICAgICAgICAgY2hhciBzekdycE5hbWVBWzQ4XTsKCiAgICAgICAgICAgICAgICAgICAgLyogRklYTUU6IE1TRE4gZG9lc24ndCBzYXkgd2hpY2ggY29kZSBwYWdlIHRoZSBXLT5BIHRyYW5zbGF0aW9uIHVzZXMsCiAgICAgICAgICAgICAgICAgICAgICogICAgICAgIG9yIHdoZXRoZXIgdGhlIGxhbmd1YWdlIG5hbWVzIGFyZSBldmVyIGxvY2FsaXNlZC4gQXNzdW1lIENQX0FDUC4KICAgICAgICAgICAgICAgICAgICAgKi8KCiAgICAgICAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIHN6TnVtYmVyLCAtMSwgc3pOdW1iZXJBLCBzaXplb2Yoc3pOdW1iZXJBKSwgMCwgMCk7CiAgICAgICAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIHN6R3JwTmFtZSwgLTEsIHN6R3JwTmFtZUEsIHNpemVvZihzekdycE5hbWVBKSwgMCwgMCk7CgogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NBKCBsZ3JwaWQsIHN6TnVtYmVyQSwgc3pHcnBOYW1lQSwgbHBQcm9jcy0+ZHdGbGFncywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbHBQcm9jcy0+bFBhcmFtICk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIHVsSW5kZXgrKzsKICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgICAgICBiQ29udGludWUgPSBGQUxTRTsKCiAgICAgICAgaWYgKCFiQ29udGludWUpCiAgICAgICAgICAgIGJyZWFrOwogICAgfQoKICAgIGlmIChoS2V5KQogICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ2FsbCBhIHVzZXJzIGZ1bmN0aW9uIGZvciBlYWNoIGxhbmd1YWdlIGdyb3VwIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIHBMYW5nR3JwRW51bVByb2MgW0ldIENhbGxiYWNrIGZ1bmN0aW9uIHRvIGNhbGwgZm9yIGVhY2ggbGFuZ3VhZ2UgZ3JvdXAKICogIGR3RmxhZ3MgICAgICAgICAgW0ldIExHUlBJRF9TVVBQT1JURUQ9QWxsIFN1cHBvcnRlZCwgTEdSUElEX0lOU1RBTExFRD1JbnN0YWxsZWQgb25seQogKiAgbFBhcmFtICAgICAgICAgICBbSV0gVXNlciBwYXJhbWV0ZXIgdG8gcGFzcyB0byBwTGFuZ0dycEVudW1Qcm9jCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBKExBTkdVQUdFR1JPVVBfRU5VTVBST0NBIHBMYW5nR3JwRW51bVByb2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNTEFOR1VBR0VHUk9VUF9DQUxMQkFDS1MgcHJvY3M7CgogICAgVFJBQ0UoIiglcCwweCUwOGxYLDB4JTA4bFgpXG4iLCBwTGFuZ0dycEVudW1Qcm9jLCBkd0ZsYWdzLCBsUGFyYW0pOwoKICAgIHByb2NzLnByb2NBID0gcExhbmdHcnBFbnVtUHJvYzsKICAgIHByb2NzLnByb2NXID0gTlVMTDsKICAgIHByb2NzLmR3RmxhZ3MgPSBkd0ZsYWdzOwogICAgcHJvY3MubFBhcmFtID0gbFBhcmFtOwoKICAgIHJldHVybiBOTFNfRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzKCBwTGFuZ0dycEVudW1Qcm9jID8gJnByb2NzIDogTlVMTCk7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc1cgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBLgogKi8KQk9PTCBXSU5BUEkgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzVyhMQU5HVUFHRUdST1VQX0VOVU1QUk9DVyBwTGFuZ0dycEVudW1Qcm9jLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTUxBTkdVQUdFR1JPVVBfQ0FMTEJBQ0tTIHByb2NzOwoKICAgIFRSQUNFKCIoJXAsMHglMDhsWCwweCUwOGxYKVxuIiwgcExhbmdHcnBFbnVtUHJvYywgZHdGbGFncywgbFBhcmFtKTsKCiAgICBwcm9jcy5wcm9jQSA9IE5VTEw7CiAgICBwcm9jcy5wcm9jVyA9IHBMYW5nR3JwRW51bVByb2M7CiAgICBwcm9jcy5kd0ZsYWdzID0gZHdGbGFnczsKICAgIHByb2NzLmxQYXJhbSA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1TeXN0ZW1MYW5ndWFnZUdyb3VwcyggcExhbmdHcnBFbnVtUHJvYyA/ICZwcm9jcyA6IE5VTEwpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc1ZhbGlkTGFuZ3VhZ2VHcm91cCAgICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgbGFuZ3VhZ2UgZ3JvdXAgaXMgc3VwcG9ydGVkIGFuZC9vciBpbnN0YWxsZWQuCiAqCiAqIFBBUkFNUwogKiAgbGdycGlkICBbSV0gTGFuZ3VhZ2UgR3JvdXAgSWQgKExHUlBJRF8gdmFsdWVzIGZyb20gIndpbm5scy5oIikKICogIGR3RmxhZ3MgW0ldIExHUlBJRF9TVVBQT1JURUQ9U3VwcG9ydGVkLCBMR1JQSURfSU5TVEFMTEVEPUluc3RhbGxlZAogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBpZiBsZ3JwaWQgaXMgc3VwcG9ydGVkIGFuZC9vciBpbnN0YWxsZWQsIGFjY29yZGluZyB0byBkd0ZsYWdzLgogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNWYWxpZExhbmd1YWdlR3JvdXAoTEdSUElEIGxncnBpZCwgRFdPUkQgZHdGbGFncykKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIHN6Rm9ybWF0W10gPSB7ICclJywneCcsJ1wwJyB9OwogICAgV0NIQVIgc3pWYWx1ZU5hbWVbMTZdLCBzelZhbHVlWzJdOwogICAgQk9PTCBiU3VwcG9ydGVkID0gRkFMU0UsIGJJbnN0YWxsZWQgPSBGQUxTRTsKICAgIEhBTkRMRSBoS2V5OwoKCiAgICBzd2l0Y2ggKGR3RmxhZ3MpCiAgICB7CiAgICBjYXNlIExHUlBJRF9JTlNUQUxMRUQ6CiAgICBjYXNlIExHUlBJRF9TVVBQT1JURUQ6CgogICAgICAgIGhLZXkgPSBOTFNfUmVnT3BlblN1YktleSggTkxTX1JlZ09wZW5LZXkoIDAsIHN6TmxzS2V5TmFtZSApLCBzekxhbmdHcm91cHNLZXlOYW1lICk7CgogICAgICAgIHNwcmludGZXKCBzelZhbHVlTmFtZSwgc3pGb3JtYXQsIGxncnBpZCApOwoKICAgICAgICBpZiAoTkxTX1JlZ0dldER3b3JkKCBoS2V5LCBzelZhbHVlTmFtZSwgKExQRFdPUkQpJnN6VmFsdWUgKSkKICAgICAgICB7CiAgICAgICAgICAgIGJTdXBwb3J0ZWQgPSBUUlVFOwoKICAgICAgICAgICAgaWYgKHN6VmFsdWVbMF0gPT0gJzEnKQogICAgICAgICAgICAgICAgYkluc3RhbGxlZCA9IFRSVUU7CiAgICAgICAgfQoKICAgICAgICBpZiAoaEtleSkKICAgICAgICAgICAgTnRDbG9zZSggaEtleSApOwoKICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAoKGR3RmxhZ3MgPT0gTEdSUElEX1NVUFBPUlRFRCAmJiBiU3VwcG9ydGVkKSB8fAogICAgICAgIChkd0ZsYWdzID09IExHUlBJRF9JTlNUQUxMRUQgJiYgYkluc3RhbGxlZCkpCiAgICAgICAgcmV0dXJuIFRSVUU7CgogICAgcmV0dXJuIEZBTFNFOwp9CgovKiBDYWxsYmFjayBmdW5jdGlvbiBwdHJzIGZvciBFbnVtTGFuZ3VhZ2VHcm91cGxvY2FsZXNBL1cgKi8KdHlwZWRlZiBzdHJ1Y3QKewogIExBTkdHUk9VUExPQ0FMRV9FTlVNUFJPQ0EgcHJvY0E7CiAgTEFOR0dST1VQTE9DQUxFX0VOVU1QUk9DVyBwcm9jVzsKICBEV09SRCAgICBkd0ZsYWdzOwogIExHUlBJRCAgIGxncnBpZDsKICBMT05HX1BUUiBsUGFyYW07Cn0gRU5VTUxBTkdVQUdFR1JPVVBMT0NBTEVfQ0FMTEJBQ0tTOwoKLyogSW50ZXJuYWwgaW1wbGVtZW50YXRpb24gb2YgRW51bUxhbmd1YWdlR3JvdXBsb2NhbGVzQS9XICovCnN0YXRpYyBCT09MIE5MU19FbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXMoRU5VTUxBTkdVQUdFR1JPVVBMT0NBTEVfQ0FMTEJBQ0tTICpscFByb2NzKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc3pMb2NhbGVLZXlOYW1lW10gPSB7CiAgICAgICdMJywnbycsJ2MnLCdhJywnbCcsJ2UnLCdcMCcKICAgIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc3pBbHRlcm5hdGVTb3J0c0tleU5hbWVbXSA9IHsKICAgICAgJ0EnLCdsJywndCcsJ2UnLCdyJywnbicsJ2EnLCd0JywnZScsJyAnLCdTJywnbycsJ3InLCd0JywncycsJ1wwJwogICAgfTsKICAgIFdDSEFSIHN6TnVtYmVyWzEwXSwgc3pWYWx1ZVs0XTsKICAgIEhBTkRMRSBoS2V5OwogICAgQk9PTCBiQ29udGludWUgPSBUUlVFLCBiQWx0ZXJuYXRlID0gRkFMU0U7CiAgICBMR1JQSUQgbGdycGlkOwogICAgVUxPTkcgdWxJbmRleCA9IDE7ICAvKiBJZ25vcmUgZGVmYXVsdCBlbnRyeSBvZiAxc3Qga2V5ICovCgogICAgaWYgKCFscFByb2NzIHx8ICFscFByb2NzLT5sZ3JwaWQgfHwgbHBQcm9jcy0+bGdycGlkID4gTEdSUElEX0FSTUVOSUFOKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmIChscFByb2NzLT5kd0ZsYWdzKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaEtleSA9IE5MU19SZWdPcGVuU3ViS2V5KCBOTFNfUmVnT3BlbktleSggMCwgc3pObHNLZXlOYW1lICksIHN6TG9jYWxlS2V5TmFtZSApOwoKICAgIGlmICghaEtleSkKICAgICAgICBXQVJOKCJOTFMgcmVnaXN0cnkga2V5IG5vdCBmb3VuZC4gUGxlYXNlIGFwcGx5IHRoZSBkZWZhdWx0IHJlZ2lzdHJ5IGZpbGUgJ3dpbmUuaW5mJ1xuIik7CgogICAgd2hpbGUgKGJDb250aW51ZSkKICAgIHsKICAgICAgICBpZiAoTkxTX1JlZ0VudW1WYWx1ZSggaEtleSwgdWxJbmRleCwgc3pOdW1iZXIsIHNpemVvZihzek51bWJlciksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN6VmFsdWUsIHNpemVvZihzelZhbHVlKSApKQogICAgICAgIHsKICAgICAgICAgICAgbGdycGlkID0gc3RydG91bFcoIHN6VmFsdWUsIE5VTEwsIDE2ICk7CgogICAgICAgICAgICBUUkFDRSgibGNpZCAlcywgZ3JwaWQgJWxkICglc21hdGNoZWQpXG4iLCBkZWJ1Z3N0cl93KHN6TnVtYmVyKSwKICAgICAgICAgICAgICAgICAgIGxncnBpZCwgbGdycGlkID09IGxwUHJvY3MtPmxncnBpZCA/ICIiIDogIm5vdCAiKTsKCiAgICAgICAgICAgIGlmIChsZ3JwaWQgPT0gbHBQcm9jcy0+bGdycGlkKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBMQ0lEIGxjaWQ7CgogICAgICAgICAgICAgICAgbGNpZCA9IHN0cnRvdWxXKCBzek51bWJlciwgTlVMTCwgMTYgKTsKCiAgICAgICAgICAgICAgICAvKiBGSVhNRTogbmF0aXZlIHJldHVybnMgZXh0cmEgdGV4dCBmb3IgYSBmZXcgKDE3LzE1MCkgbG9jYWxlcywgZS5nOgogICAgICAgICAgICAgICAgICogJzAwMDAwNDM3ICAgICAgICAgIDtHZW9yZ2lhbicKICAgICAgICAgICAgICAgICAqIEF0IHByZXNlbnQgd2Ugb25seSBwYXNzIHRoZSBMQ0lEIHN0cmluZy4KICAgICAgICAgICAgICAgICAqLwoKICAgICAgICAgICAgICAgIGlmIChscFByb2NzLT5wcm9jVykKICAgICAgICAgICAgICAgICAgICBiQ29udGludWUgPSBscFByb2NzLT5wcm9jVyggbGdycGlkLCBsY2lkLCBzek51bWJlciwgbHBQcm9jcy0+bFBhcmFtICk7CiAgICAgICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgY2hhciBzek51bWJlckFbc2l6ZW9mKHN6TnVtYmVyKS9zaXplb2YoV0NIQVIpXTsKCiAgICAgICAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIHN6TnVtYmVyLCAtMSwgc3pOdW1iZXJBLCBzaXplb2Yoc3pOdW1iZXJBKSwgMCwgMCk7CgogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NBKCBsZ3JwaWQsIGxjaWQsIHN6TnVtYmVyQSwgbHBQcm9jcy0+bFBhcmFtICk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIHVsSW5kZXgrKzsKICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgIHsKICAgICAgICAgICAgLyogRmluaXNoZWQgZW51bWVyYXRpbmcgdGhpcyBrZXkgKi8KICAgICAgICAgICAgaWYgKCFiQWx0ZXJuYXRlKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAvKiBFbnVtZXJhdGUgYWx0ZXJuYXRlIHNvcnRzIGFsc28gKi8KICAgICAgICAgICAgICAgIGhLZXkgPSBOTFNfUmVnT3BlbktleSggaEtleSwgc3pBbHRlcm5hdGVTb3J0c0tleU5hbWUgKTsKICAgICAgICAgICAgICAgIGJBbHRlcm5hdGUgPSBUUlVFOwogICAgICAgICAgICAgICAgdWxJbmRleCA9IDA7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZQogICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gRkFMU0U7IC8qIEZpbmlzaGVkIGJvdGgga2V5cyAqLwogICAgICAgIH0KCiAgICAgICAgaWYgKCFiQ29udGludWUpCiAgICAgICAgICAgIGJyZWFrOwogICAgfQoKICAgIGlmIChoS2V5KQogICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ2FsbCBhIHVzZXJzIGZ1bmN0aW9uIGZvciBldmVyeSBsb2NhbGUgaW4gYSBsYW5ndWFnZSBncm91cCBhdmFpbGFibGUgb24gdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBwTGFuZ0dycExjRW51bVByb2MgW0ldIENhbGxiYWNrIGZ1bmN0aW9uIHRvIGNhbGwgZm9yIGVhY2ggbG9jYWxlCiAqICBsZ3JwaWQgICAgICAgICAgICAgW0ldIExhbmd1YWdlIGdyb3VwIChMR1JQSURfIHZhbHVlcyBmcm9tICJ3aW5ubHMuaCIpCiAqICBkd0ZsYWdzICAgICAgICAgICAgW0ldIFJlc2VydmVkLCBzZXQgdG8gMAogKiAgbFBhcmFtICAgICAgICAgICAgIFtJXSBVc2VyIHBhcmFtZXRlciB0byBwYXNzIHRvIHBMYW5nR3JwTGNFbnVtUHJvYwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KQk9PTCBXSU5BUEkgRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzQShMQU5HR1JPVVBMT0NBTEVfRU5VTVBST0NBIHBMYW5nR3JwTGNFbnVtUHJvYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMR1JQSUQgbGdycGlkLCBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1MQU5HVUFHRUdST1VQTE9DQUxFX0NBTExCQUNLUyBjYWxsYmFja3M7CgogICAgVFJBQ0UoIiglcCwweCUwOGxYLDB4JTA4bFgsMHglMDhsWClcbiIsIHBMYW5nR3JwTGNFbnVtUHJvYywgbGdycGlkLCBkd0ZsYWdzLCBsUGFyYW0pOwoKICAgIGNhbGxiYWNrcy5wcm9jQSAgID0gcExhbmdHcnBMY0VudW1Qcm9jOwogICAgY2FsbGJhY2tzLnByb2NXICAgPSBOVUxMOwogICAgY2FsbGJhY2tzLmR3RmxhZ3MgPSBkd0ZsYWdzOwogICAgY2FsbGJhY2tzLmxncnBpZCAgPSBsZ3JwaWQ7CiAgICBjYWxsYmFja3MubFBhcmFtICA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1MYW5ndWFnZUdyb3VwTG9jYWxlcyggcExhbmdHcnBMY0VudW1Qcm9jID8gJmNhbGxiYWNrcyA6IE5VTEwgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEVudW1MYW5ndWFnZUdyb3VwTG9jYWxlc0EuCiAqLwpCT09MIFdJTkFQSSBFbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXNXKExBTkdHUk9VUExPQ0FMRV9FTlVNUFJPQ1cgcExhbmdHcnBMY0VudW1Qcm9jLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExHUlBJRCBsZ3JwaWQsIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTUxBTkdVQUdFR1JPVVBMT0NBTEVfQ0FMTEJBQ0tTIGNhbGxiYWNrczsKCiAgICBUUkFDRSgiKCVwLDB4JTA4bFgsMHglMDhsWCwweCUwOGxYKVxuIiwgcExhbmdHcnBMY0VudW1Qcm9jLCBsZ3JwaWQsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgY2FsbGJhY2tzLnByb2NBICAgPSBOVUxMOwogICAgY2FsbGJhY2tzLnByb2NXICAgPSBwTGFuZ0dycExjRW51bVByb2M7CiAgICBjYWxsYmFja3MuZHdGbGFncyA9IGR3RmxhZ3M7CiAgICBjYWxsYmFja3MubGdycGlkICA9IGxncnBpZDsKICAgIGNhbGxiYWNrcy5sUGFyYW0gID0gbFBhcmFtOwoKICAgIHJldHVybiBOTFNfRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzKCBwTGFuZ0dycExjRW51bVByb2MgPyAmY2FsbGJhY2tzIDogTlVMTCApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtR2VvSUQgICAgKEtFUk5FTDMyLkApCiAqCiAqIENhbGwgYSB1c2VycyBmdW5jdGlvbiBmb3IgZXZlcnkgbG9jYXRpb24gYXZhaWxhYmxlIG9uIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgZ2VvY2xhc3MgICAgIFtJXSBUeXBlIG9mIGluZm9ybWF0aW9uIGRlc2lyZWQgKFNZU0dFT1RZUEUgZW51bSBmcm9tICJ3aW5ubHMuaCIpCiAqICByZXNlcnZlZCAgICAgW0ldIFJlc2VydmVkLCBzZXQgdG8gMAogKiAgcEdlb0VudW1Qcm9jIFtJXSBDYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIGZvciBlYWNoIGxvY2F0aW9uCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtR2VvSUQoR0VPQ0xBU1MgZ2VvY2xhc3MsIEdFT0lEIHJlc2VydmVkLCBHRU9fRU5VTVBST0MgcEdlb0VudW1Qcm9jKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc3pDb3VudHJ5Q29kZVZhbHVlTmFtZVtdID0gewogICAgICAnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsJ0MnLCdvJywnZCcsJ2UnLCdcMCcKICAgIH07CiAgICBXQ0hBUiBzek51bWJlclsxMF07CiAgICBIQU5ETEUgaEtleTsKICAgIFVMT05HIHVsSW5kZXggPSAwOwoKICAgIFRSQUNFKCIoMHglMDhsWCwweCUwOGxYLCVwKVxuIiwgZ2VvY2xhc3MsIHJlc2VydmVkLCBwR2VvRW51bVByb2MpOwoKICAgIGlmIChnZW9jbGFzcyAhPSBHRU9DTEFTU19OQVRJT04gfHwgcmVzZXJ2ZWQgfHwgIXBHZW9FbnVtUHJvYykKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBoS2V5ID0gTkxTX1JlZ09wZW5LZXkoIDAsIHN6Q291bnRyeUxpc3ROYW1lICk7CgogICAgd2hpbGUgKE5MU19SZWdFbnVtU3ViS2V5KCBoS2V5LCB1bEluZGV4LCBzek51bWJlciwgc2l6ZW9mKHN6TnVtYmVyKSApKQogICAgewogICAgICAgIEJPT0wgYkNvbnRpbnVlID0gVFJVRTsKICAgICAgICBEV09SRCBkd0dlb0lkOwogICAgICAgIEhBTkRMRSBoU3ViS2V5ID0gTkxTX1JlZ09wZW5LZXkoIGhLZXksIHN6TnVtYmVyICk7CgogICAgICAgIGlmIChoU3ViS2V5KQogICAgICAgIHsKICAgICAgICAgICAgaWYgKE5MU19SZWdHZXREd29yZCggaFN1YktleSwgc3pDb3VudHJ5Q29kZVZhbHVlTmFtZSwgJmR3R2VvSWQgKSkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgVFJBQ0UoIkdvdCBnZW9pZCAlbGRcbiIsIGR3R2VvSWQpOwoKICAgICAgICAgICAgICAgIGlmICghcEdlb0VudW1Qcm9jKCBkd0dlb0lkICkpCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gRkFMU0U7CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIE50Q2xvc2UoIGhTdWJLZXkgKTsKICAgICAgICB9CgogICAgICAgIGlmICghYkNvbnRpbnVlKQogICAgICAgICAgICBicmVhazsKCiAgICAgICAgdWxJbmRleCsrOwogICAgfQoKICAgIGlmIChoS2V5KQogICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSW52YWxpZGF0ZU5MU0NhY2hlICAgICAgICAgICAoS0VSTkVMMzIuQCkKICoKICogSW52YWxpZGF0ZSB0aGUgY2FjaGUgb2YgTkxTIHZhbHVlcy4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLgogKiAgRmFpbHVyZTogRkFMU0UuCiAqLwpCT09MIFdJTkFQSSBJbnZhbGlkYXRlTkxTQ2FjaGUodm9pZCkKewogIEZJWE1FKCIoKSBzdHViXG4iKTsKICByZXR1cm4gRkFMU0U7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFVzZXJHZW9JRCAoS0VSTkVMMzIuQCkKICovCkdFT0lEIFdJTkFQSSBHZXRVc2VyR2VvSUQoIEdFT0NMQVNTIEdlb0NsYXNzICkKewogICAgRklYTUUoIiVsZFxuIixHZW9DbGFzcyk7CiAgICByZXR1cm4gR0VPSURfTk9UX0FWQUlMQUJMRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgU2V0VXNlckdlb0lEIChLRVJORUwzMi5AKQogKi8KQk9PTCBXSU5BUEkgU2V0VXNlckdlb0lEKCBHRU9JRCBHZW9JRCApCnsKICAgIEZJWE1FKCIlbGRcbiIsR2VvSUQpOwogICAgcmV0dXJuIEZBTFNFOwp9Cgp0eXBlZGVmIHN0cnVjdAp7CiAgICB1bmlvbgogICAgewogICAgICAgIFVJTEFOR1VBR0VfRU5VTVBST0NBIHByb2NBOwogICAgICAgIFVJTEFOR1VBR0VfRU5VTVBST0NXIHByb2NXOwogICAgfSB1OwogICAgRFdPUkQgZmxhZ3M7CiAgICBMT05HX1BUUiBwYXJhbTsKfSBFTlVNX1VJTEFOR19DQUxMQkFDSzsKCnN0YXRpYyBCT09MIENBTExCQUNLIGVudW1fdWlsYW5nX3Byb2NfYSggSE1PRFVMRSBoTW9kdWxlLCBMUENTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExPTkdfUFRSIGxQYXJhbSApCnsKICAgIEVOVU1fVUlMQU5HX0NBTExCQUNLICplbnVtX3VpbGFuZyA9IChFTlVNX1VJTEFOR19DQUxMQkFDSyAqKWxQYXJhbTsKICAgIGNoYXIgYnVmWzIwXTsKCiAgICBzcHJpbnRmKGJ1ZiwgIiUwOHgiLCAoVUlOVClMYW5nSUQpOwogICAgcmV0dXJuIGVudW1fdWlsYW5nLT51LnByb2NBKCBidWYsIGVudW1fdWlsYW5nLT5wYXJhbSApOwp9CgpzdGF0aWMgQk9PTCBDQUxMQkFDSyBlbnVtX3VpbGFuZ19wcm9jX3coIEhNT0RVTEUgaE1vZHVsZSwgTFBDV1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1dTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExPTkdfUFRSIGxQYXJhbSApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBmb3JtYXRXW10gPSB7JyUnLCcwJywnOCcsJ3gnLDB9OwogICAgRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgKmVudW1fdWlsYW5nID0gKEVOVU1fVUlMQU5HX0NBTExCQUNLICopbFBhcmFtOwogICAgV0NIQVIgYnVmWzIwXTsKCiAgICBzcHJpbnRmVyggYnVmLCBmb3JtYXRXLCAoVUlOVClMYW5nSUQgKTsKICAgIHJldHVybiBlbnVtX3VpbGFuZy0+dS5wcm9jVyggYnVmLCBlbnVtX3VpbGFuZy0+cGFyYW0gKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVVJTGFuZ3VhZ2VzQSAoS0VSTkVMMzIuQCkKICovCkJPT0wgV0lOQVBJIEVudW1VSUxhbmd1YWdlc0EoVUlMQU5HVUFHRV9FTlVNUFJPQ0EgcFVJTGFuZ0VudW1Qcm9jLCBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1fVUlMQU5HX0NBTExCQUNLIGVudW1fdWlsYW5nOwoKICAgIFRSQUNFKCIlcCwgJWx4LCAlbHhcbiIsIHBVSUxhbmdFbnVtUHJvYywgZHdGbGFncywgbFBhcmFtKTsKCiAgICBpZighcFVJTGFuZ0VudW1Qcm9jKSB7CglTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQogICAgaWYoZHdGbGFncykgewoJU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGVudW1fdWlsYW5nLnUucHJvY0EgPSBwVUlMYW5nRW51bVByb2M7CiAgICBlbnVtX3VpbGFuZy5mbGFncyA9IGR3RmxhZ3M7CiAgICBlbnVtX3VpbGFuZy5wYXJhbSA9IGxQYXJhbTsKCiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNBKCBrZXJuZWwzMl9oYW5kbGUsIChMUENTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQQ1NUUilMT0NBTEVfSUxBTkdVQUdFLCBlbnVtX3VpbGFuZ19wcm9jX2EsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTE9OR19QVFIpJmVudW1fdWlsYW5nKTsKICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtVUlMYW5ndWFnZXNXIChLRVJORUwzMi5AKQogKi8KQk9PTCBXSU5BUEkgRW51bVVJTGFuZ3VhZ2VzVyhVSUxBTkdVQUdFX0VOVU1QUk9DVyBwVUlMYW5nRW51bVByb2MsIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgZW51bV91aWxhbmc7CgogICAgVFJBQ0UoIiVwLCAlbHgsICVseFxuIiwgcFVJTGFuZ0VudW1Qcm9jLCBkd0ZsYWdzLCBsUGFyYW0pOwoKCiAgICBpZighcFVJTGFuZ0VudW1Qcm9jKSB7CglTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQogICAgaWYoZHdGbGFncykgewoJU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGVudW1fdWlsYW5nLnUucHJvY1cgPSBwVUlMYW5nRW51bVByb2M7CiAgICBlbnVtX3VpbGFuZy5mbGFncyA9IGR3RmxhZ3M7CiAgICBlbnVtX3VpbGFuZy5wYXJhbSA9IGxQYXJhbTsKCiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNXKCBrZXJuZWwzMl9oYW5kbGUsIChMUENXU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIGVudW1fdWlsYW5nX3Byb2NfdywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMT05HX1BUUikmZW51bV91aWxhbmcpOwogICAgcmV0dXJuIFRSVUU7Cn0KCklOVCBXSU5BUEkgR2V0R2VvSW5mb1coR0VPSUQgR2VvSWQsIEdFT1RZUEUgR2VvVHlwZSwgTFBXU1RSIGxwR2VvRGF0YSwgCiAgICAgICAgICAgICAgICBpbnQgY2NoRGF0YSwgTEFOR0lEIGxhbmd1YWdlKQp7CiAgICBGSVhNRSgiJWxkICVsZCAlcCAlZCAlZFxuIiwgR2VvSWQsIEdlb1R5cGUsIGxwR2VvRGF0YSwgY2NoRGF0YSwgbGFuZ3VhZ2UpOwogICAgcmV0dXJuIDA7Cn0KCklOVCBXSU5BUEkgR2V0R2VvSW5mb0EoR0VPSUQgR2VvSWQsIEdFT1RZUEUgR2VvVHlwZSwgTFBTVFIgbHBHZW9EYXRhLCAKICAgICAgICAgICAgICAgIGludCBjY2hEYXRhLCBMQU5HSUQgbGFuZ3VhZ2UpCnsKICAgIEZJWE1FKCIlbGQgJWxkICVwICVkICVkXG4iLCBHZW9JZCwgR2VvVHlwZSwgbHBHZW9EYXRhLCBjY2hEYXRhLCBsYW5ndWFnZSk7CiAgICByZXR1cm4gMDsKfQo=