LyoKICogTG9jYWxlIHN1cHBvcnQKICoKICogQ29weXJpZ2h0IDE5OTUgTWFydGluIHZvbiBMb2V3aXMKICogQ29weXJpZ2h0IDE5OTggRGF2aWQgTGVlIExhbWJlcnQKICogQ29weXJpZ2h0IDIwMDAgSnVsaW8gQ+lzYXIgR+F6cXVlegogKiBDb3B5cmlnaHQgMjAwMiBBbGV4YW5kcmUgSnVsbGlhcmQgZm9yIENvZGVXZWF2ZXJzCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKICogbW9kaWZ5IGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIKICogdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlIEdOVQogKiBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYWxvbmcgd2l0aCB0aGlzIGxpYnJhcnk7IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKICogRm91bmRhdGlvbiwgSW5jLiwgNTEgRnJhbmtsaW4gU3QsIEZpZnRoIEZsb29yLCBCb3N0b24sIE1BIDAyMTEwLTEzMDEsIFVTQQogKi8KCiNpbmNsdWRlICJjb25maWcuaCIKI2luY2x1ZGUgIndpbmUvcG9ydC5oIgoKI2luY2x1ZGUgPGFzc2VydC5oPgojaW5jbHVkZSA8bG9jYWxlLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KI2luY2x1ZGUgPHN0ZGFyZy5oPgojaW5jbHVkZSA8c3RkaW8uaD4KI2luY2x1ZGUgPGN0eXBlLmg+CiNpbmNsdWRlIDxzdGRsaWIuaD4KCiNpZmRlZiBfX0FQUExFX18KIyBpbmNsdWRlIDxDb3JlRm91bmRhdGlvbi9DRkJ1bmRsZS5oPgojIGluY2x1ZGUgPENvcmVGb3VuZGF0aW9uL0NGTG9jYWxlLmg+CiMgaW5jbHVkZSA8Q29yZUZvdW5kYXRpb24vQ0ZTdHJpbmcuaD4KI2VuZGlmCgojaW5jbHVkZSAibnRzdGF0dXMuaCIKI2RlZmluZSBXSU4zMl9OT19TVEFUVVMKI2luY2x1ZGUgIndpbmRlZi5oIgojaW5jbHVkZSAid2luYmFzZS5oIgojaW5jbHVkZSAid2ludXNlci5oIiAgLyogZm9yIFJUX1NUUklOR1cgKi8KI2luY2x1ZGUgIndpbnRlcm5sLmgiCiNpbmNsdWRlICJ3aW5lL3VuaWNvZGUuaCIKI2luY2x1ZGUgIndpbm5scy5oIgojaW5jbHVkZSAid2luZXJyb3IuaCIKI2luY2x1ZGUgIndpbnZlci5oIgojaW5jbHVkZSAia2VybmVsX3ByaXZhdGUuaCIKI2luY2x1ZGUgIndpbmUvZGVidWcuaCIKCldJTkVfREVGQVVMVF9ERUJVR19DSEFOTkVMKG5scyk7CgojZGVmaW5lIExPQ0FMRV9MT0NBTEVJTkZPRkxBR1NNQVNLIChMT0NBTEVfTk9VU0VST1ZFUlJJREV8TE9DQUxFX1VTRV9DUF9BQ1B8TE9DQUxFX1JFVFVSTl9OVU1CRVIpCgovKiBjdXJyZW50IGNvZGUgcGFnZXMgKi8Kc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKmFuc2lfY3B0YWJsZTsKc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKm9lbV9jcHRhYmxlOwpzdGF0aWMgY29uc3QgdW5pb24gY3B0YWJsZSAqbWFjX2NwdGFibGU7CnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICp1bml4X2NwdGFibGU7ICAvKiBOVUxMIGlmIFVURjggKi8KCnN0YXRpYyBIQU5ETEUgTkxTX1JlZ09wZW5LZXkoSEFORExFIGhSb290S2V5LCBMUENXU1RSIHN6S2V5TmFtZSk7CgpzdGF0aWMgY29uc3QgV0NIQVIgc3pObHNLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1wwJwp9OwoKc3RhdGljIGNvbnN0IFdDSEFSIHN6TG9jYWxlS2V5TmFtZVtdID0gewogICAgJ00nLCdhJywnYycsJ2gnLCdpJywnbicsJ2UnLCdcXCcsJ1MnLCd5JywncycsJ3QnLCdlJywnbScsJ1xcJywKICAgICdDJywndScsJ3InLCdyJywnZScsJ24nLCd0JywnQycsJ28nLCduJywndCcsJ3InLCdvJywnbCcsJ1MnLCdlJywndCcsJ1xcJywKICAgICdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnXFwnLCdOJywnbCcsJ3MnLCdcXCcsJ0wnLCdvJywnYycsJ2EnLCdsJywnZScsMAp9OwoKc3RhdGljIGNvbnN0IFdDSEFSIHN6Q29kZXBhZ2VLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1xcJywnQycsJ28nLCdkJywnZScsJ3AnLCdhJywnZycsJ2UnLDAKfTsKCnN0YXRpYyBjb25zdCBXQ0hBUiBzekxhbmdHcm91cHNLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1xcJywKICAgICdMJywnYScsJ24nLCdnJywndScsJ2EnLCdnJywnZScsJyAnLCdHJywncicsJ28nLCd1JywncCcsJ3MnLDAKfTsKCi8qIENoYXJzZXQgdG8gY29kZXBhZ2UgbWFwLCBzb3J0ZWQgYnkgbmFtZS4gKi8Kc3RhdGljIGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5CnsKICAgIGNvbnN0IGNoYXIgKmNoYXJzZXRfbmFtZTsKICAgIFVJTlQgICAgICAgIGNvZGVwYWdlOwp9IGNoYXJzZXRfbmFtZXNbXSA9CnsKICAgIHsgIkJJRzUiLCA5NTAgfSwKICAgIHsgIkNQMTI1MCIsIDEyNTAgfSwKICAgIHsgIkNQMTI1MSIsIDEyNTEgfSwKICAgIHsgIkNQMTI1MiIsIDEyNTIgfSwKICAgIHsgIkNQMTI1MyIsIDEyNTMgfSwKICAgIHsgIkNQMTI1NCIsIDEyNTQgfSwKICAgIHsgIkNQMTI1NSIsIDEyNTUgfSwKICAgIHsgIkNQMTI1NiIsIDEyNTYgfSwKICAgIHsgIkNQMTI1NyIsIDEyNTcgfSwKICAgIHsgIkNQMTI1OCIsIDEyNTggfSwKICAgIHsgIkNQOTMyIiwgOTMyIH0sCiAgICB7ICJDUDkzNiIsIDkzNiB9LAogICAgeyAiQ1A5NDkiLCA5NDkgfSwKICAgIHsgIkNQOTUwIiwgOTUwIH0sCiAgICB7ICJFVUNKUCIsIDIwOTMyIH0sCiAgICB7ICJHQjIzMTIiLCA5MzYgfSwKICAgIHsgIklCTTAzNyIsIDM3IH0sCiAgICB7ICJJQk0xMDI2IiwgMTAyNiB9LAogICAgeyAiSUJNNDI0IiwgNDI0IH0sCiAgICB7ICJJQk00MzciLCA0MzcgfSwKICAgIHsgIklCTTUwMCIsIDUwMCB9LAogICAgeyAiSUJNODUwIiwgODUwIH0sCiAgICB7ICJJQk04NTIiLCA4NTIgfSwKICAgIHsgIklCTTg1NSIsIDg1NSB9LAogICAgeyAiSUJNODU3IiwgODU3IH0sCiAgICB7ICJJQk04NjAiLCA4NjAgfSwKICAgIHsgIklCTTg2MSIsIDg2MSB9LAogICAgeyAiSUJNODYyIiwgODYyIH0sCiAgICB7ICJJQk04NjMiLCA4NjMgfSwKICAgIHsgIklCTTg2NCIsIDg2NCB9LAogICAgeyAiSUJNODY1IiwgODY1IH0sCiAgICB7ICJJQk04NjYiLCA4NjYgfSwKICAgIHsgIklCTTg2OSIsIDg2OSB9LAogICAgeyAiSUJNODc0IiwgODc0IH0sCiAgICB7ICJJQk04NzUiLCA4NzUgfSwKICAgIHsgIklTTzg4NTkxIiwgMjg1OTEgfSwKICAgIHsgIklTTzg4NTkxMCIsIDI4NjAwIH0sCiAgICB7ICJJU084ODU5MTMiLCAyODYwMyB9LAogICAgeyAiSVNPODg1OTE0IiwgMjg2MDQgfSwKICAgIHsgIklTTzg4NTkxNSIsIDI4NjA1IH0sCiAgICB7ICJJU084ODU5MTYiLCAyODYwNiB9LAogICAgeyAiSVNPODg1OTIiLCAyODU5MiB9LAogICAgeyAiSVNPODg1OTMiLCAyODU5MyB9LAogICAgeyAiSVNPODg1OTQiLCAyODU5NCB9LAogICAgeyAiSVNPODg1OTUiLCAyODU5NSB9LAogICAgeyAiSVNPODg1OTYiLCAyODU5NiB9LAogICAgeyAiSVNPODg1OTciLCAyODU5NyB9LAogICAgeyAiSVNPODg1OTgiLCAyODU5OCB9LAogICAgeyAiSVNPODg1OTkiLCAyODU5OSB9LAogICAgeyAiS09JOFIiLCAyMDg2NiB9LAogICAgeyAiS09JOFUiLCAyMTg2NiB9LAogICAgeyAiVVRGOCIsIENQX1VURjggfQp9OwoKCnN0cnVjdCBsb2NhbGVfbmFtZQp7CiAgICBXQ0hBUiAgd2luX25hbWVbMTI4XTsgICAvKiBXaW5kb3dzIG5hbWUgKCJlbi1VUyIpICovCiAgICBXQ0hBUiAgbGFuZ1sxMjhdOyAgICAgICAvKiBsYW5ndWFnZSAoImVuIikgKG5vdGU6IGJ1ZmZlciBjb250YWlucyB0aGUgb3RoZXIgc3RyaW5ncyB0b28pICovCiAgICBXQ0hBUiAqY291bnRyeTsgICAgICAgICAvKiBjb3VudHJ5ICgiVVMiKSAqLwogICAgV0NIQVIgKmNoYXJzZXQ7ICAgICAgICAgLyogY2hhcnNldCAoIlVURi04IikgZm9yIFVuaXggZm9ybWF0IG9ubHkgKi8KICAgIFdDSEFSICpzY3JpcHQ7ICAgICAgICAgIC8qIHNjcmlwdCAoIkxhdG4iKSBmb3IgV2luZG93cyBmb3JtYXQgb25seSAqLwogICAgV0NIQVIgKm1vZGlmaWVyOyAgICAgICAgLyogbW9kaWZpZXIgb3Igc29ydCBvcmRlciAqLwogICAgTENJRCAgIGxjaWQ7ICAgICAgICAgICAgLyogY29ycmVzcG9uZGluZyBMQ0lEICovCiAgICBpbnQgICAgbWF0Y2hlczsgICAgICAgICAvKiBudW1iZXIgb2YgZWxlbWVudHMgbWF0Y2hpbmcgTENJRCAoMC4uNCkgKi8KICAgIFVJTlQgICBjb2RlcGFnZTsgICAgICAgIC8qIGNvZGVwYWdlIGNvcnJlc3BvbmRpbmcgdG8gY2hhcnNldCAqLwp9OwoKLyogbG9jYWxlIGlkcyBjb3JyZXNwb25kaW5nIHRvIHRoZSB2YXJpb3VzIFVuaXggbG9jYWxlIHBhcmFtZXRlcnMgKi8Kc3RhdGljIExDSUQgbGNpZF9MQ19DT0xMQVRFOwpzdGF0aWMgTENJRCBsY2lkX0xDX0NUWVBFOwpzdGF0aWMgTENJRCBsY2lkX0xDX01FU1NBR0VTOwpzdGF0aWMgTENJRCBsY2lkX0xDX01PTkVUQVJZOwpzdGF0aWMgTENJRCBsY2lkX0xDX05VTUVSSUM7CnN0YXRpYyBMQ0lEIGxjaWRfTENfVElNRTsKc3RhdGljIExDSUQgbGNpZF9MQ19QQVBFUjsKc3RhdGljIExDSUQgbGNpZF9MQ19NRUFTVVJFTUVOVDsKc3RhdGljIExDSUQgbGNpZF9MQ19URUxFUEhPTkU7CgovKiBDb3B5IEFzY2lpIHN0cmluZyB0byBVbmljb2RlIHdpdGhvdXQgdXNpbmcgY29kZXBhZ2VzICovCnN0YXRpYyBpbmxpbmUgdm9pZCBzdHJjcHluQXRvVyggV0NIQVIgKmRzdCwgY29uc3QgY2hhciAqc3JjLCBzaXplX3QgbiApCnsKICAgIHdoaWxlIChuID4gMSAmJiAqc3JjKQogICAgewogICAgICAgICpkc3QrKyA9ICh1bnNpZ25lZCBjaGFyKSpzcmMrKzsKICAgICAgICBuLS07CiAgICB9CiAgICBpZiAobikgKmRzdCA9IDA7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWdldF9sY2lkX2NvZGVwYWdlCiAqCiAqIFJldHJpZXZlIHRoZSBBTlNJIGNvZGVwYWdlIGZvciBhIGdpdmVuIGxvY2FsZS4KICovCnN0YXRpYyBpbmxpbmUgVUlOVCBnZXRfbGNpZF9jb2RlcGFnZSggTENJRCBsY2lkICkKewogICAgVUlOVCByZXQ7CiAgICBpZiAoIUdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfSURFRkFVTFRBTlNJQ09ERVBBR0V8TE9DQUxFX1JFVFVSTl9OVU1CRVIsIChXQ0hBUiAqKSZyZXQsCiAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YocmV0KS9zaXplb2YoV0NIQVIpICkpIHJldCA9IDA7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfY29kZXBhZ2VfdGFibGUKICoKICogRmluZCB0aGUgdGFibGUgZm9yIGEgZ2l2ZW4gY29kZXBhZ2UsIGhhbmRsaW5nIENQX0FDUCBldGMuIHBzZXVkby1jb2RlcGFnZXMKICovCnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICpnZXRfY29kZXBhZ2VfdGFibGUoIHVuc2lnbmVkIGludCBjb2RlcGFnZSApCnsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnJldCA9IE5VTEw7CgogICAgYXNzZXJ0KCBhbnNpX2NwdGFibGUgKTsgIC8qIGluaXQgbXVzdCBoYXZlIGJlZW4gZG9uZSBhbHJlYWR5ICovCgogICAgc3dpdGNoKGNvZGVwYWdlKQogICAgewogICAgY2FzZSBDUF9BQ1A6CiAgICAgICAgcmV0dXJuIGFuc2lfY3B0YWJsZTsKICAgIGNhc2UgQ1BfT0VNQ1A6CiAgICAgICAgcmV0dXJuIG9lbV9jcHRhYmxlOwogICAgY2FzZSBDUF9NQUNDUDoKICAgICAgICByZXR1cm4gbWFjX2NwdGFibGU7CiAgICBjYXNlIENQX1VURjc6CiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENQX1RIUkVBRF9BQ1A6CiAgICAgICAgaWYgKCEoY29kZXBhZ2UgPSBrZXJuZWxfZ2V0X3RocmVhZF9kYXRhKCktPmNvZGVfcGFnZSkpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBkZWZhdWx0OgogICAgICAgIGlmIChjb2RlcGFnZSA9PSBhbnNpX2NwdGFibGUtPmluZm8uY29kZXBhZ2UpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG9lbV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gb2VtX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG1hY19jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gbWFjX2NwdGFibGU7CiAgICAgICAgcmV0ID0gd2luZV9jcF9nZXRfdGFibGUoIGNvZGVwYWdlICk7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBjaGFyc2V0X2NtcCAoaW50ZXJuYWwpCiAqLwpzdGF0aWMgaW50IGNoYXJzZXRfY21wKCBjb25zdCB2b2lkICpuYW1lLCBjb25zdCB2b2lkICplbnRyeSApCnsKICAgIGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICpjaGFyc2V0ID0gKGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICopZW50cnk7CiAgICByZXR1cm4gc3RyY2FzZWNtcCggKGNvbnN0IGNoYXIgKiluYW1lLCBjaGFyc2V0LT5jaGFyc2V0X25hbWUgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlmaW5kX2NoYXJzZXQKICovCnN0YXRpYyBVSU5UIGZpbmRfY2hhcnNldCggY29uc3QgV0NIQVIgKm5hbWUgKQp7CiAgICBjb25zdCBzdHJ1Y3QgY2hhcnNldF9lbnRyeSAqZW50cnk7CiAgICBjaGFyIGNoYXJzZXRfbmFtZVsxNl07CiAgICBzaXplX3QgaSwgajsKCiAgICAvKiByZW1vdmUgcHVuY3R1YXRpb24gY2hhcmFjdGVycyBmcm9tIGNoYXJzZXQgbmFtZSAqLwogICAgZm9yIChpID0gaiA9IDA7IG5hbWVbaV0gJiYgaiA8IHNpemVvZihjaGFyc2V0X25hbWUpLTE7IGkrKykKICAgICAgICBpZiAoaXNhbG51bSgodW5zaWduZWQgY2hhciluYW1lW2ldKSkgY2hhcnNldF9uYW1lW2orK10gPSBuYW1lW2ldOwogICAgY2hhcnNldF9uYW1lW2pdID0gMDsKCiAgICBlbnRyeSA9IGJzZWFyY2goIGNoYXJzZXRfbmFtZSwgY2hhcnNldF9uYW1lcywKICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXMpL3NpemVvZihjaGFyc2V0X25hbWVzWzBdKSwKICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXNbMF0pLCBjaGFyc2V0X2NtcCApOwogICAgaWYgKGVudHJ5KSByZXR1cm4gZW50cnktPmNvZGVwYWdlOwogICAgcmV0dXJuIDA7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGZpbmRfbG9jYWxlX2lkX2NhbGxiYWNrCiAqLwpzdGF0aWMgQk9PTCBDQUxMQkFDSyBmaW5kX2xvY2FsZV9pZF9jYWxsYmFjayggSE1PRFVMRSBoTW9kdWxlLCBMUENXU1RSIHR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENXU1RSIG5hbWUsIFdPUkQgTGFuZ0lELCBMUEFSQU0gbFBhcmFtICkKewogICAgc3RydWN0IGxvY2FsZV9uYW1lICpkYXRhID0gKHN0cnVjdCBsb2NhbGVfbmFtZSAqKWxQYXJhbTsKICAgIFdDSEFSIGJ1ZmZlclsxMjhdOwogICAgaW50IG1hdGNoZXMgPSAwOwogICAgTENJRCBsY2lkID0gTUFLRUxDSUQoIExhbmdJRCwgU09SVF9ERUZBVUxUICk7ICAvKiBGSVhNRTogaGFuZGxlIHNvcnQgb3JkZXIgKi8KCiAgICBpZiAoUFJJTUFSWUxBTkdJRChMYW5nSUQpID09IExBTkdfTkVVVFJBTCkgcmV0dXJuIFRSVUU7IC8qIGNvbnRpbnVlIHNlYXJjaCAqLwoKICAgIC8qIGZpcnN0IGNoZWNrIGV4YWN0IG5hbWUgKi8KICAgIGlmIChkYXRhLT53aW5fbmFtZVswXSAmJgogICAgICAgIEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU05BTUUgfCBMT0NBTEVfTk9VU0VST1ZFUlJJREUsCiAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZmZlciwgc2l6ZW9mKGJ1ZmZlcikvc2l6ZW9mKFdDSEFSKSApKQogICAgewogICAgICAgIGlmICghc3RyY21wVyggZGF0YS0+d2luX25hbWUsIGJ1ZmZlciApKQogICAgICAgIHsKICAgICAgICAgICAgbWF0Y2hlcyA9IDQ7ICAvKiBldmVyeXRoaW5nIG1hdGNoZXMgKi8KICAgICAgICAgICAgZ290byBkb25lOwogICAgICAgIH0KICAgIH0KCiAgICBpZiAoIUdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU0lTTzYzOUxBTkdOQU1FIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLAogICAgICAgICAgICAgICAgICAgICAgICAgYnVmZmVyLCBzaXplb2YoYnVmZmVyKS9zaXplb2YoV0NIQVIpICkpCiAgICAgICAgcmV0dXJuIFRSVUU7CiAgICBpZiAoc3RyY21wVyggYnVmZmVyLCBkYXRhLT5sYW5nICkpIHJldHVybiBUUlVFOwogICAgbWF0Y2hlcysrOyAgLyogbGFuZ3VhZ2UgbmFtZSBtYXRjaGVkICovCgogICAgaWYgKGRhdGEtPmNvdW50cnkpCiAgICB7CiAgICAgICAgaWYgKEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU0lTTzMxNjZDVFJZTkFNRXxMT0NBTEVfTk9VU0VST1ZFUlJJREUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBidWZmZXIsIHNpemVvZihidWZmZXIpL3NpemVvZihXQ0hBUikgKSkKICAgICAgICB7CiAgICAgICAgICAgIGlmIChzdHJjbXBXKCBidWZmZXIsIGRhdGEtPmNvdW50cnkgKSkgZ290byBkb25lOwogICAgICAgICAgICBtYXRjaGVzKys7ICAvKiBjb3VudHJ5IG5hbWUgbWF0Y2hlZCAqLwogICAgICAgIH0KICAgIH0KICAgIGVsc2UgIC8qIG1hdGNoIGRlZmF1bHQgbGFuZ3VhZ2UgKi8KICAgIHsKICAgICAgICBpZiAoU1VCTEFOR0lEKExhbmdJRCkgPT0gU1VCTEFOR19ERUZBVUxUKSBtYXRjaGVzKys7CiAgICB9CgogICAgaWYgKGRhdGEtPmNvZGVwYWdlKQogICAgewogICAgICAgIFVJTlQgdW5peF9jcDsKICAgICAgICBpZiAoR2V0TG9jYWxlSW5mb1coIGxjaWQsIExPQ0FMRV9JREVGQVVMVFVOSVhDT0RFUEFHRSB8IExPQ0FMRV9SRVRVUk5fTlVNQkVSLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQV1NUUikmdW5peF9jcCwgc2l6ZW9mKHVuaXhfY3ApL3NpemVvZihXQ0hBUikgKSkKICAgICAgICB7CiAgICAgICAgICAgIGlmICh1bml4X2NwID09IGRhdGEtPmNvZGVwYWdlKSBtYXRjaGVzKys7CiAgICAgICAgfQogICAgfQoKICAgIC8qIEZJWE1FOiBjaGVjayBzb3J0IG9yZGVyICovCgpkb25lOgogICAgaWYgKG1hdGNoZXMgPiBkYXRhLT5tYXRjaGVzKQogICAgewogICAgICAgIGRhdGEtPmxjaWQgPSBsY2lkOwogICAgICAgIGRhdGEtPm1hdGNoZXMgPSBtYXRjaGVzOwogICAgfQogICAgcmV0dXJuIChkYXRhLT5tYXRjaGVzIDwgNCk7ICAvKiBubyBuZWVkIHRvIGNvbnRpbnVlIGZvciBwZXJmZWN0IG1hdGNoICovCn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCXBhcnNlX2xvY2FsZV9uYW1lCiAqCiAqIFBhcnNlIGEgbG9jYWxlIG5hbWUgaW50byBhIHN0cnVjdCBsb2NhbGVfbmFtZSwgaGFuZGxpbmcgYm90aCBXaW5kb3dzIGFuZCBVbml4IGZvcm1hdHMuCiAqIFVuaXggZm9ybWF0IGlzOiBsYW5nW19jb3VudHJ5XVsuY2hhcnNldF1bQG1vZGlmaWVyXQogKiBXaW5kb3dzIGZvcm1hdCBpczogbGFuZ1stc2NyaXB0XVstY291bnRyeV1bX21vZGlmaWVyXQogKi8Kc3RhdGljIHZvaWQgcGFyc2VfbG9jYWxlX25hbWUoIGNvbnN0IFdDSEFSICpzdHIsIHN0cnVjdCBsb2NhbGVfbmFtZSAqbmFtZSApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzZXBXW10gPSB7Jy0nLCdfJywnLicsJ0AnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHdpbnNlcFdbXSA9IHsnLScsJ18nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHBvc2l4V1tdID0geydQJywnTycsJ1MnLCdJJywnWCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgY1dbXSA9IHsnQycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGF0aW5XW10gPSB7J2wnLCdhJywndCcsJ2knLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsYXRuV1tdID0geyctJywnTCcsJ2EnLCd0JywnbicsMH07CiAgICBXQ0hBUiAqcDsKCiAgICBUUkFDRSgiJXNcbiIsIGRlYnVnc3RyX3coc3RyKSk7CgogICAgbmFtZS0+Y291bnRyeSA9IG5hbWUtPmNoYXJzZXQgPSBuYW1lLT5zY3JpcHQgPSBuYW1lLT5tb2RpZmllciA9IE5VTEw7CiAgICBuYW1lLT5sY2lkID0gTUFLRUxDSUQoIE1BS0VMQU5HSUQoTEFOR19FTkdMSVNILFNVQkxBTkdfREVGQVVMVCksIFNPUlRfREVGQVVMVCApOwogICAgbmFtZS0+bWF0Y2hlcyA9IDA7CiAgICBuYW1lLT5jb2RlcGFnZSA9IDA7CiAgICBuYW1lLT53aW5fbmFtZVswXSA9IDA7CiAgICBsc3RyY3B5blcoIG5hbWUtPmxhbmcsIHN0ciwgc2l6ZW9mKG5hbWUtPmxhbmcpL3NpemVvZihXQ0hBUikgKTsKCiAgICBpZiAoIShwID0gc3RycGJya1coIG5hbWUtPmxhbmcsIHNlcFcgKSkpCiAgICB7CiAgICAgICAgaWYgKCFzdHJjbXBXKCBuYW1lLT5sYW5nLCBwb3NpeFcgKSB8fCAhc3RyY21wVyggbmFtZS0+bGFuZywgY1cgKSkKICAgICAgICB7CiAgICAgICAgICAgIG5hbWUtPm1hdGNoZXMgPSA0OyAgLyogcGVyZmVjdCBtYXRjaCBmb3IgZGVmYXVsdCBFbmdsaXNoIGxjaWQgKi8KICAgICAgICAgICAgcmV0dXJuOwogICAgICAgIH0KICAgICAgICBzdHJjcHlXKCBuYW1lLT53aW5fbmFtZSwgbmFtZS0+bGFuZyApOwogICAgfQogICAgZWxzZSBpZiAoKnAgPT0gJy0nKSAgLyogV2luZG93cyBmb3JtYXQgKi8KICAgIHsKICAgICAgICBzdHJjcHlXKCBuYW1lLT53aW5fbmFtZSwgbmFtZS0+bGFuZyApOwogICAgICAgICpwKysgPSAwOwogICAgICAgIG5hbWUtPmNvdW50cnkgPSBwOwogICAgICAgIGlmICghKHAgPSBzdHJwYnJrVyggcCwgd2luc2VwVyApKSkgZ290byBkb25lOwogICAgICAgIGlmICgqcCA9PSAnLScpCiAgICAgICAgewogICAgICAgICAgICAqcCsrID0gMDsKICAgICAgICAgICAgbmFtZS0+c2NyaXB0ID0gbmFtZS0+Y291bnRyeTsKICAgICAgICAgICAgbmFtZS0+Y291bnRyeSA9IHA7CiAgICAgICAgICAgIGlmICghKHAgPSBzdHJwYnJrVyggcCwgd2luc2VwVyApKSkgZ290byBkb25lOwogICAgICAgIH0KICAgICAgICAqcCsrID0gMDsKICAgICAgICBuYW1lLT5tb2RpZmllciA9IHA7CiAgICB9CiAgICBlbHNlICAvKiBVbml4IGZvcm1hdCAqLwogICAgewogICAgICAgIGlmICgqcCA9PSAnXycpCiAgICAgICAgewogICAgICAgICAgICAqcCsrID0gMDsKICAgICAgICAgICAgbmFtZS0+Y291bnRyeSA9IHA7CiAgICAgICAgICAgIHAgPSBzdHJwYnJrVyggcCwgc2VwVyArIDIgKTsKICAgICAgICB9CiAgICAgICAgaWYgKHAgJiYgKnAgPT0gJy4nKQogICAgICAgIHsKICAgICAgICAgICAgKnArKyA9IDA7CiAgICAgICAgICAgIG5hbWUtPmNoYXJzZXQgPSBwOwogICAgICAgICAgICBwID0gc3RyY2hyVyggcCwgJ0AnICk7CiAgICAgICAgfQogICAgICAgIGlmIChwKQogICAgICAgIHsKICAgICAgICAgICAgKnArKyA9IDA7CiAgICAgICAgICAgIG5hbWUtPm1vZGlmaWVyID0gcDsKICAgICAgICB9CgogICAgICAgIGlmIChuYW1lLT5jaGFyc2V0KQogICAgICAgICAgICBuYW1lLT5jb2RlcGFnZSA9IGZpbmRfY2hhcnNldCggbmFtZS0+Y2hhcnNldCApOwoKICAgICAgICAvKiByZWJ1aWxkIGEgV2luZG93cyBuYW1lIGlmIHBvc3NpYmxlICovCgogICAgICAgIGlmIChuYW1lLT5jaGFyc2V0KSBnb3RvIGRvbmU7ICAvKiBjYW4ndCBzcGVjaWZ5IGNoYXJzZXQgaW4gV2luZG93cyBmb3JtYXQgKi8KICAgICAgICBpZiAobmFtZS0+bW9kaWZpZXIgJiYgc3RyY21wVyggbmFtZS0+bW9kaWZpZXIsIGxhdGluVyApKQogICAgICAgICAgICBnb3RvIGRvbmU7ICAvKiBvbmx5IExhdG4gc2NyaXB0IHN1cHBvcnRlZCBmb3Igbm93ICovCiAgICAgICAgc3RyY3B5VyggbmFtZS0+d2luX25hbWUsIG5hbWUtPmxhbmcgKTsKICAgICAgICBpZiAobmFtZS0+bW9kaWZpZXIpIHN0cmNhdFcoIG5hbWUtPndpbl9uYW1lLCBsYXRuVyApOwogICAgICAgIGlmIChuYW1lLT5jb3VudHJ5KQogICAgICAgIHsKICAgICAgICAgICAgcCA9IG5hbWUtPndpbl9uYW1lICsgc3RybGVuVyhuYW1lLT53aW5fbmFtZSk7CiAgICAgICAgICAgICpwKysgPSAnLSc7CiAgICAgICAgICAgIHN0cmNweVcoIHAsIG5hbWUtPmNvdW50cnkgKTsKICAgICAgICB9CiAgICB9CmRvbmU6CiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNXKCBrZXJuZWwzMl9oYW5kbGUsIChMUENXU1RSKVJUX1NUUklORywgKExQQ1dTVFIpTE9DQUxFX0lMQU5HVUFHRSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpbmRfbG9jYWxlX2lkX2NhbGxiYWNrLCAoTFBBUkFNKW5hbWUgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgY29udmVydF9kZWZhdWx0X2xjaWQKICoKICogR2V0IHRoZSBkZWZhdWx0IExDSUQgdG8gdXNlIGZvciBhIGdpdmVuIGxjdHlwZSBpbiBHZXRMb2NhbGVJbmZvLgogKi8Kc3RhdGljIExDSUQgY29udmVydF9kZWZhdWx0X2xjaWQoIExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSApCnsKICAgIGlmIChsY2lkID09IExPQ0FMRV9TWVNURU1fREVGQVVMVCB8fAogICAgICAgIGxjaWQgPT0gTE9DQUxFX1VTRVJfREVGQVVMVCB8fAogICAgICAgIGxjaWQgPT0gTE9DQUxFX05FVVRSQUwpCiAgICB7CiAgICAgICAgTENJRCBkZWZhdWx0X2lkID0gMDsKCiAgICAgICAgc3dpdGNoKGxjdHlwZSAmIDB4ZmZmZikKICAgICAgICB7CiAgICAgICAgY2FzZSBMT0NBTEVfU1NPUlROQU1FOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19DT0xMQVRFOwogICAgICAgICAgICBicmVhazsKCiAgICAgICAgY2FzZSBMT0NBTEVfRk9OVFNJR05BVFVSRToKICAgICAgICBjYXNlIExPQ0FMRV9JREVGQVVMVEFOU0lDT0RFUEFHRToKICAgICAgICBjYXNlIExPQ0FMRV9JREVGQVVMVENPREVQQUdFOgogICAgICAgIGNhc2UgTE9DQUxFX0lERUZBVUxURUJDRElDQ09ERVBBR0U6CiAgICAgICAgY2FzZSBMT0NBTEVfSURFRkFVTFRNQUNDT0RFUEFHRToKICAgICAgICBjYXNlIExPQ0FMRV9JREVGQVVMVFVOSVhDT0RFUEFHRToKICAgICAgICAgICAgZGVmYXVsdF9pZCA9IGxjaWRfTENfQ1RZUEU7CiAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICBjYXNlIExPQ0FMRV9JQ1VSUkRJR0lUUzoKICAgICAgICBjYXNlIExPQ0FMRV9JQ1VSUkVOQ1k6CiAgICAgICAgY2FzZSBMT0NBTEVfSUlOVExDVVJSRElHSVRTOgogICAgICAgIGNhc2UgTE9DQUxFX0lORUdDVVJSOgogICAgICAgIGNhc2UgTE9DQUxFX0lORUdTRVBCWVNQQUNFOgogICAgICAgIGNhc2UgTE9DQUxFX0lORUdTSUdOUE9TTjoKICAgICAgICBjYXNlIExPQ0FMRV9JTkVHU1lNUFJFQ0VERVM6CiAgICAgICAgY2FzZSBMT0NBTEVfSVBPU1NFUEJZU1BBQ0U6CiAgICAgICAgY2FzZSBMT0NBTEVfSVBPU1NJR05QT1NOOgogICAgICAgIGNhc2UgTE9DQUxFX0lQT1NTWU1QUkVDRURFUzoKICAgICAgICBjYXNlIExPQ0FMRV9TQ1VSUkVOQ1k6CiAgICAgICAgY2FzZSBMT0NBTEVfU0lOVExTWU1CT0w6CiAgICAgICAgY2FzZSBMT0NBTEVfU01PTkRFQ0lNQUxTRVA6CiAgICAgICAgY2FzZSBMT0NBTEVfU01PTkdST1VQSU5HOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE9VU0FORFNFUDoKICAgICAgICBjYXNlIExPQ0FMRV9TTkFUSVZFQ1VSUk5BTUU6CiAgICAgICAgICAgIGRlZmF1bHRfaWQgPSBsY2lkX0xDX01PTkVUQVJZOwogICAgICAgICAgICBicmVhazsKCiAgICAgICAgY2FzZSBMT0NBTEVfSURJR0lUUzoKICAgICAgICBjYXNlIExPQ0FMRV9JRElHSVRTVUJTVElUVVRJT046CiAgICAgICAgY2FzZSBMT0NBTEVfSUxaRVJPOgogICAgICAgIGNhc2UgTE9DQUxFX0lORUdOVU1CRVI6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RFQ0lNQUw6CiAgICAgICAgY2FzZSBMT0NBTEVfU0dST1VQSU5HOgogICAgICAgIGNhc2UgTE9DQUxFX1NOQU46CiAgICAgICAgY2FzZSBMT0NBTEVfU05BVElWRURJR0lUUzoKICAgICAgICBjYXNlIExPQ0FMRV9TTkVHQVRJVkVTSUdOOgogICAgICAgIGNhc2UgTE9DQUxFX1NORUdJTkZJTklUWToKICAgICAgICBjYXNlIExPQ0FMRV9TUE9TSU5GSU5JVFk6CiAgICAgICAgY2FzZSBMT0NBTEVfU1BPU0lUSVZFU0lHTjoKICAgICAgICBjYXNlIExPQ0FMRV9TVEhPVVNBTkQ6CiAgICAgICAgICAgIGRlZmF1bHRfaWQgPSBsY2lkX0xDX05VTUVSSUM7CiAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICBjYXNlIExPQ0FMRV9JQ0FMRU5EQVJUWVBFOgogICAgICAgIGNhc2UgTE9DQUxFX0lDRU5UVVJZOgogICAgICAgIGNhc2UgTE9DQUxFX0lEQVRFOgogICAgICAgIGNhc2UgTE9DQUxFX0lEQVlMWkVSTzoKICAgICAgICBjYXNlIExPQ0FMRV9JRklSU1REQVlPRldFRUs6CiAgICAgICAgY2FzZSBMT0NBTEVfSUZJUlNUV0VFS09GWUVBUjoKICAgICAgICBjYXNlIExPQ0FMRV9JTERBVEU6CiAgICAgICAgY2FzZSBMT0NBTEVfSU1PTkxaRVJPOgogICAgICAgIGNhc2UgTE9DQUxFX0lPUFRJT05BTENBTEVOREFSOgogICAgICAgIGNhc2UgTE9DQUxFX0lUSU1FOgogICAgICAgIGNhc2UgTE9DQUxFX0lUSU1FTUFSS1BPU046CiAgICAgICAgY2FzZSBMT0NBTEVfSVRMWkVSTzoKICAgICAgICBjYXNlIExPQ0FMRV9TMTE1OToKICAgICAgICBjYXNlIExPQ0FMRV9TMjM1OToKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWREFZTkFNRTE6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVkRBWU5BTUUyOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZEQVlOQU1FMzoKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWREFZTkFNRTQ6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVkRBWU5BTUU1OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZEQVlOQU1FNjoKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWREFZTkFNRTc6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTE6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTI6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTM6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTQ6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTU6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTY6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTc6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTg6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTk6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTEwOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUxMToKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWTU9OVEhOQU1FMTI6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTEzOgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVRFOgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVlOQU1FMToKICAgICAgICBjYXNlIExPQ0FMRV9TREFZTkFNRTI6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RBWU5BTUUzOgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVlOQU1FNDoKICAgICAgICBjYXNlIExPQ0FMRV9TREFZTkFNRTU6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RBWU5BTUU2OgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVlOQU1FNzoKICAgICAgICBjYXNlIExPQ0FMRV9TRFVSQVRJT046CiAgICAgICAgY2FzZSBMT0NBTEVfU0xPTkdEQVRFOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUxOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUyOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUzOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU0OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU1OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU2OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU3OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU4OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUU5OgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUxMDoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMTE6CiAgICAgICAgY2FzZSBMT0NBTEVfU01PTlRITkFNRTEyOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUxMzoKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlREQVRFOgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUUxOgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUUyOgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUUzOgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUU0OgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUU1OgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUU2OgogICAgICAgIGNhc2UgTE9DQUxFX1NTSE9SVEVTVERBWU5BTUU3OgogICAgICAgIGNhc2UgTE9DQUxFX1NUSU1FOgogICAgICAgIGNhc2UgTE9DQUxFX1NUSU1FRk9STUFUOgogICAgICAgIGNhc2UgTE9DQUxFX1NZRUFSTU9OVEg6CiAgICAgICAgICAgIGRlZmF1bHRfaWQgPSBsY2lkX0xDX1RJTUU7CiAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICBjYXNlIExPQ0FMRV9JUEFQRVJTSVpFOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19QQVBFUjsKICAgICAgICAgICAgYnJlYWs7CgogICAgICAgIGNhc2UgTE9DQUxFX0lNRUFTVVJFOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19NRUFTVVJFTUVOVDsKICAgICAgICAgICAgYnJlYWs7CgogICAgICAgIGNhc2UgTE9DQUxFX0lDT1VOVFJZOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19URUxFUEhPTkU7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KICAgICAgICBpZiAoZGVmYXVsdF9pZCkgbGNpZCA9IGRlZmF1bHRfaWQ7CiAgICB9CiAgICByZXR1cm4gQ29udmVydERlZmF1bHRMb2NhbGUoIGxjaWQgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJY3JlYXRlX3JlZ2lzdHJ5X2tleQogKgogKiBDcmVhdGUgdGhlIENvbnRyb2wgUGFuZWxcXEludGVybmF0aW9uYWwgcmVnaXN0cnkga2V5LgogKi8Kc3RhdGljIGlubGluZSBIQU5ETEUgY3JlYXRlX3JlZ2lzdHJ5X2tleSh2b2lkKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaW50bFdbXSA9IHsnQycsJ28nLCduJywndCcsJ3InLCdvJywnbCcsJyAnLCdQJywnYScsJ24nLCdlJywnbCcsJ1xcJywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdJJywnbicsJ3QnLCdlJywncicsJ24nLCdhJywndCcsJ2knLCdvJywnbicsJ2EnLCdsJywwfTsKICAgIE9CSkVDVF9BVFRSSUJVVEVTIGF0dHI7CiAgICBVTklDT0RFX1NUUklORyBuYW1lVzsKICAgIEhBTkRMRSBoa2V5OwoKICAgIGlmIChSdGxPcGVuQ3VycmVudFVzZXIoIEtFWV9BTExfQUNDRVNTLCAmaGtleSApICE9IFNUQVRVU19TVUNDRVNTKSByZXR1cm4gMDsKCiAgICBhdHRyLkxlbmd0aCA9IHNpemVvZihhdHRyKTsKICAgIGF0dHIuUm9vdERpcmVjdG9yeSA9IGhrZXk7CiAgICBhdHRyLk9iamVjdE5hbWUgPSAmbmFtZVc7CiAgICBhdHRyLkF0dHJpYnV0ZXMgPSAwOwogICAgYXR0ci5TZWN1cml0eURlc2NyaXB0b3IgPSBOVUxMOwogICAgYXR0ci5TZWN1cml0eVF1YWxpdHlPZlNlcnZpY2UgPSBOVUxMOwogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgaW50bFcgKTsKCiAgICBpZiAoTnRDcmVhdGVLZXkoICZoa2V5LCBLRVlfQUxMX0FDQ0VTUywgJmF0dHIsIDAsIE5VTEwsIDAsIE5VTEwgKSAhPSBTVEFUVVNfU1VDQ0VTUykgaGtleSA9IDA7CiAgICBOdENsb3NlKCBhdHRyLlJvb3REaXJlY3RvcnkgKTsKICAgIHJldHVybiBoa2V5Owp9CgoKLyogdXBkYXRlIHRoZSByZWdpc3RyeSBzZXR0aW5ncyBmb3IgYSBnaXZlbiBsb2NhbGUgcGFyYW1ldGVyICovCi8qIHJldHVybiBUUlVFIGlmIGFuIHVwZGF0ZSB3YXMgbmVlZGVkICovCnN0YXRpYyBCT09MIGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIEhLRVkgaGtleSwgY29uc3QgV0NIQVIgKm5hbWUsIExDSUQgbGNpZCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgTENUWVBFICp2YWx1ZXMsIFVJTlQgbmJfdmFsdWVzICkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGZvcm1hdFdbXSA9IHsgJyUnLCcwJywnOCcsJ3gnLDAgfTsKICAgIFdDSEFSIGJ1ZmZlcldbNDBdOwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVc7CiAgICBEV09SRCBjb3VudCwgaTsKCiAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJm5hbWVXLCBuYW1lICk7CiAgICBjb3VudCA9IHNpemVvZihidWZmZXJXKTsKICAgIGlmICghTnRRdWVyeVZhbHVlS2V5KGhrZXksICZuYW1lVywgS2V5VmFsdWVQYXJ0aWFsSW5mb3JtYXRpb24sIChMUEJZVEUpYnVmZmVyVywgY291bnQsICZjb3VudCkpCiAgICB7CiAgICAgICAgY29uc3QgS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKmluZm8gPSAoS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKilidWZmZXJXOwogICAgICAgIExQQ1dTVFIgdGV4dCA9IChMUENXU1RSKWluZm8tPkRhdGE7CgogICAgICAgIGlmIChzdHJ0b3VsVyggdGV4dCwgTlVMTCwgMTYgKSA9PSBsY2lkKSByZXR1cm4gRkFMU0U7IC8qIGFscmVhZHkgc2V0IGNvcnJlY3RseSAqLwogICAgICAgIFRSQUNFKCAidXBkYXRpbmcgcmVnaXN0cnksIGxvY2FsZSAlcyBjaGFuZ2VkICVzIC0+ICUwOHhcbiIsCiAgICAgICAgICAgICAgIGRlYnVnc3RyX3cobmFtZSksIGRlYnVnc3RyX3codGV4dCksIGxjaWQgKTsKICAgIH0KICAgIGVsc2UgVFJBQ0UoICJ1cGRhdGluZyByZWdpc3RyeSwgbG9jYWxlICVzIGNoYW5nZWQgbm9uZSAtPiAlMDh4XG4iLCBkZWJ1Z3N0cl93KG5hbWUpLCBsY2lkICk7CiAgICBzcHJpbnRmVyggYnVmZmVyVywgZm9ybWF0VywgbGNpZCApOwogICAgTnRTZXRWYWx1ZUtleSggaGtleSwgJm5hbWVXLCAwLCBSRUdfU1osIGJ1ZmZlclcsIChzdHJsZW5XKGJ1ZmZlclcpICsgMSkgKiBzaXplb2YoV0NIQVIpICk7CgogICAgZm9yIChpID0gMDsgaSA8IG5iX3ZhbHVlczsgaSsrKQogICAgewogICAgICAgIEdldExvY2FsZUluZm9XKCBsY2lkLCB2YWx1ZXNbaV0gfCBMT0NBTEVfTk9VU0VST1ZFUlJJREUsIGJ1ZmZlclcsCiAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihidWZmZXJXKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgU2V0TG9jYWxlSW5mb1coIGxjaWQsIHZhbHVlc1tpXSwgYnVmZmVyVyApOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUxPQ0FMRV9Jbml0UmVnaXN0cnkKICoKICogVXBkYXRlIHJlZ2lzdHJ5IGNvbnRlbnRzIG9uIHN0YXJ0dXAgaWYgdGhlIHVzZXIgbG9jYWxlIGhhcyBjaGFuZ2VkLgogKiBUaGlzIHNpbXVsYXRlcyB0aGUgYWN0aW9uIG9mIHRoZSBXaW5kb3dzIGNvbnRyb2wgcGFuZWwuCiAqLwp2b2lkIExPQ0FMRV9Jbml0UmVnaXN0cnkodm9pZCkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGFjcFdbXSA9IHsnQScsJ0MnLCdQJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBvZW1jcFdbXSA9IHsnTycsJ0UnLCdNJywnQycsJ1AnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIG1hY2NwV1tdID0geydNJywnQScsJ0MnLCdDJywnUCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbG9jYWxlV1tdID0geydMJywnbycsJ2MnLCdhJywnbCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGxjX2N0eXBlV1tdID0geyAnTCcsJ0MnLCdfJywnQycsJ1QnLCdZJywnUCcsJ0UnLDAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY19tb25ldGFyeVdbXSA9IHsgJ0wnLCdDJywnXycsJ00nLCdPJywnTicsJ0UnLCdUJywnQScsJ1InLCdZJywwIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGNfbnVtZXJpY1dbXSA9IHsgJ0wnLCdDJywnXycsJ04nLCdVJywnTScsJ0UnLCdSJywnSScsJ0MnLDAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY190aW1lV1tdID0geyAnTCcsJ0MnLCdfJywnVCcsJ0knLCdNJywnRScsMCB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGxjX21lYXN1cmVtZW50V1tdID0geyAnTCcsJ0MnLCdfJywnTScsJ0UnLCdBJywnUycsJ1UnLCdSJywnRScsJ00nLCdFJywnTicsJ1QnLDAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY190ZWxlcGhvbmVXW10gPSB7ICdMJywnQycsJ18nLCdUJywnRScsJ0wnLCdFJywnUCcsJ0gnLCdPJywnTicsJ0UnLDAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY19wYXBlcldbXSA9IHsgJ0wnLCdDJywnXycsJ1AnLCdBJywnUCcsJ0UnLCdSJywwfTsKICAgIHN0YXRpYyBjb25zdCBzdHJ1Y3QKICAgIHsKICAgICAgICBMUENXU1RSIG5hbWU7CiAgICAgICAgVVNIT1JUIHZhbHVlOwogICAgfSB1cGRhdGVfY3BfdmFsdWVzW10gPSB7CiAgICAgICAgeyBhY3BXLCBMT0NBTEVfSURFRkFVTFRBTlNJQ09ERVBBR0UgfSwKICAgICAgICB7IG9lbWNwVywgTE9DQUxFX0lERUZBVUxUQ09ERVBBR0UgfSwKICAgICAgICB7IG1hY2NwVywgTE9DQUxFX0lERUZBVUxUTUFDQ09ERVBBR0UgfQogICAgfTsKICAgIHN0YXRpYyBjb25zdCBMQ1RZUEUgbGNfbWVzc2FnZXNfdmFsdWVzW10gPSB7CiAgICAgIExPQ0FMRV9TTEFOR1VBR0UsCiAgICAgIExPQ0FMRV9TQ09VTlRSWSwKICAgICAgTE9DQUxFX1NMSVNUIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX21vbmV0YXJ5X3ZhbHVlc1tdID0gewogICAgICBMT0NBTEVfU0NVUlJFTkNZLAogICAgICBMT0NBTEVfSUNVUlJFTkNZLAogICAgICBMT0NBTEVfSU5FR0NVUlIsCiAgICAgIExPQ0FMRV9JQ1VSUkRJR0lUUywKICAgICAgTE9DQUxFX0lMWkVSTywKICAgICAgTE9DQUxFX1NNT05ERUNJTUFMU0VQLAogICAgICBMT0NBTEVfU01PTkdST1VQSU5HLAogICAgICBMT0NBTEVfU01PTlRIT1VTQU5EU0VQIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX251bWVyaWNfdmFsdWVzW10gPSB7CiAgICAgIExPQ0FMRV9TREVDSU1BTCwKICAgICAgTE9DQUxFX1NUSE9VU0FORCwKICAgICAgTE9DQUxFX0lESUdJVFMsCiAgICAgIExPQ0FMRV9JRElHSVRTVUJTVElUVVRJT04sCiAgICAgIExPQ0FMRV9TTkFUSVZFRElHSVRTLAogICAgICBMT0NBTEVfSU5FR05VTUJFUiwKICAgICAgTE9DQUxFX1NORUdBVElWRVNJR04sCiAgICAgIExPQ0FMRV9TUE9TSVRJVkVTSUdOLAogICAgICBMT0NBTEVfU0dST1VQSU5HIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX3RpbWVfdmFsdWVzW10gPSB7CiAgICAgIExPQ0FMRV9TMTE1OSwKICAgICAgTE9DQUxFX1MyMzU5LAogICAgICBMT0NBTEVfU1RJTUUsCiAgICAgIExPQ0FMRV9JVElNRSwKICAgICAgTE9DQUxFX0lUTFpFUk8sCiAgICAgIExPQ0FMRV9TU0hPUlREQVRFLAogICAgICBMT0NBTEVfU0xPTkdEQVRFLAogICAgICBMT0NBTEVfU0RBVEUsCiAgICAgIExPQ0FMRV9JVElNRU1BUktQT1NOLAogICAgICBMT0NBTEVfSUNBTEVOREFSVFlQRSwKICAgICAgTE9DQUxFX0lGSVJTVERBWU9GV0VFSywKICAgICAgTE9DQUxFX0lGSVJTVFdFRUtPRllFQVIsCiAgICAgIExPQ0FMRV9TVElNRUZPUk1BVCwKICAgICAgTE9DQUxFX1NZRUFSTU9OVEgsCiAgICAgIExPQ0FMRV9JREFURSB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY19tZWFzdXJlbWVudF92YWx1ZXNbXSA9IHsgTE9DQUxFX0lNRUFTVVJFIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX3RlbGVwaG9uZV92YWx1ZXNbXSA9IHsgTE9DQUxFX0lDT1VOVFJZIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX3BhcGVyX3ZhbHVlc1tdID0geyBMT0NBTEVfSVBBUEVSU0laRSB9OwoKICAgIFVOSUNPREVfU1RSSU5HIG5hbWVXOwogICAgV0NIQVIgYnVmZmVyV1s4MF07CiAgICBEV09SRCBjb3VudCwgaTsKICAgIEhBTkRMRSBoa2V5OwogICAgTENJRCBsY2lkID0gR2V0VXNlckRlZmF1bHRMQ0lEKCk7CgogICAgaWYgKCEoaGtleSA9IGNyZWF0ZV9yZWdpc3RyeV9rZXkoKSkpCiAgICAgICAgcmV0dXJuOyAgLyogZG9uJ3QgZG8gYW55dGhpbmcgaWYgd2UgY2FuJ3QgY3JlYXRlIHRoZSByZWdpc3RyeSBrZXkgKi8KCiAgICBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBoa2V5LCBsb2NhbGVXLCBsY2lkX0xDX01FU1NBR0VTLCBsY19tZXNzYWdlc192YWx1ZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YobGNfbWVzc2FnZXNfdmFsdWVzKS9zaXplb2YobGNfbWVzc2FnZXNfdmFsdWVzWzBdKSApOwogICAgbG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbGNfbW9uZXRhcnlXLCBsY2lkX0xDX01PTkVUQVJZLCBsY19tb25ldGFyeV92YWx1ZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YobGNfbW9uZXRhcnlfdmFsdWVzKS9zaXplb2YobGNfbW9uZXRhcnlfdmFsdWVzWzBdKSApOwogICAgbG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbGNfbnVtZXJpY1csIGxjaWRfTENfTlVNRVJJQywgbGNfbnVtZXJpY192YWx1ZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YobGNfbnVtZXJpY192YWx1ZXMpL3NpemVvZihsY19udW1lcmljX3ZhbHVlc1swXSkgKTsKICAgIGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIGhrZXksIGxjX3RpbWVXLCBsY2lkX0xDX1RJTUUsIGxjX3RpbWVfdmFsdWVzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGxjX3RpbWVfdmFsdWVzKS9zaXplb2YobGNfdGltZV92YWx1ZXNbMF0pICk7CiAgICBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBoa2V5LCBsY19tZWFzdXJlbWVudFcsIGxjaWRfTENfTUVBU1VSRU1FTlQsIGxjX21lYXN1cmVtZW50X3ZhbHVlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihsY19tZWFzdXJlbWVudF92YWx1ZXMpL3NpemVvZihsY19tZWFzdXJlbWVudF92YWx1ZXNbMF0pICk7CiAgICBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBoa2V5LCBsY190ZWxlcGhvbmVXLCBsY2lkX0xDX1RFTEVQSE9ORSwgbGNfdGVsZXBob25lX3ZhbHVlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihsY190ZWxlcGhvbmVfdmFsdWVzKS9zaXplb2YobGNfdGVsZXBob25lX3ZhbHVlc1swXSkgKTsKICAgIGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIGhrZXksIGxjX3BhcGVyVywgbGNpZF9MQ19QQVBFUiwgbGNfcGFwZXJfdmFsdWVzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGxjX3BhcGVyX3ZhbHVlcykvc2l6ZW9mKGxjX3BhcGVyX3ZhbHVlc1swXSkgKTsKCiAgICBpZiAobG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbGNfY3R5cGVXLCBsY2lkX0xDX0NUWVBFLCBOVUxMLCAwICkpCiAgICB7CiAgICAgICAgSEtFWSBubHNfa2V5ID0gTkxTX1JlZ09wZW5LZXkoIDAsIHN6Q29kZXBhZ2VLZXlOYW1lICk7CgogICAgICAgIGZvciAoaSA9IDA7IGkgPCBzaXplb2YodXBkYXRlX2NwX3ZhbHVlcykvc2l6ZW9mKHVwZGF0ZV9jcF92YWx1ZXNbMF0pOyBpKyspCiAgICAgICAgewogICAgICAgICAgICBjb3VudCA9IEdldExvY2FsZUluZm9XKCBsY2lkLCB1cGRhdGVfY3BfdmFsdWVzW2ldLnZhbHVlIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBidWZmZXJXLCBzaXplb2YoYnVmZmVyVykvc2l6ZW9mKFdDSEFSKSApOwogICAgICAgICAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJm5hbWVXLCB1cGRhdGVfY3BfdmFsdWVzW2ldLm5hbWUgKTsKICAgICAgICAgICAgTnRTZXRWYWx1ZUtleSggbmxzX2tleSwgJm5hbWVXLCAwLCBSRUdfU1osIGJ1ZmZlclcsIGNvdW50ICogc2l6ZW9mKFdDSEFSKSApOwogICAgICAgIH0KICAgICAgICBOdENsb3NlKCBubHNfa2V5ICk7CiAgICB9CgogICAgTnRDbG9zZSggaGtleSApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBzZXR1cF91bml4X2xvY2FsZXMKICovCnN0YXRpYyBVSU5UIHNldHVwX3VuaXhfbG9jYWxlcyh2b2lkKQp7CiAgICBzdHJ1Y3QgbG9jYWxlX25hbWUgbG9jYWxlX25hbWU7CiAgICBXQ0hBUiBidWZmZXJbMTI4XSwgY3R5cGVfYnVmZlsxMjhdOwogICAgY2hhciAqbG9jYWxlOwogICAgVUlOVCB1bml4X2NwID0gMDsKCiAgICBpZiAoKGxvY2FsZSA9IHNldGxvY2FsZSggTENfQ1RZUEUsIE5VTEwgKSkpCiAgICB7CiAgICAgICAgc3RyY3B5bkF0b1coIGN0eXBlX2J1ZmYsIGxvY2FsZSwgc2l6ZW9mKGN0eXBlX2J1ZmYpL3NpemVvZihXQ0hBUikgKTsKICAgICAgICBwYXJzZV9sb2NhbGVfbmFtZSggY3R5cGVfYnVmZiwgJmxvY2FsZV9uYW1lICk7CiAgICAgICAgbGNpZF9MQ19DVFlQRSA9IGxvY2FsZV9uYW1lLmxjaWQ7CiAgICAgICAgdW5peF9jcCA9IGxvY2FsZV9uYW1lLmNvZGVwYWdlOwogICAgfQogICAgaWYgKCFsY2lkX0xDX0NUWVBFKSAgLyogdGhpcyBvbmUgbmVlZHMgYSBkZWZhdWx0IHZhbHVlICovCiAgICAgICAgbGNpZF9MQ19DVFlQRSA9IE1BS0VMQ0lEKCBNQUtFTEFOR0lEKExBTkdfRU5HTElTSCxTVUJMQU5HX0RFRkFVTFQpLCBTT1JUX0RFRkFVTFQgKTsKCiAgICBUUkFDRSggImdvdCBsY2lkICUwNHggKCVkIG1hdGNoZXMpIGZvciBMQ19DVFlQRT0lc1xuIiwKICAgICAgICAgICBsb2NhbGVfbmFtZS5sY2lkLCBsb2NhbGVfbmFtZS5tYXRjaGVzLCBkZWJ1Z3N0cl9hKGxvY2FsZSkgKTsKCiNkZWZpbmUgR0VUX1VOSVhfTE9DQUxFKGNhdCkgZG8gXAogICAgaWYgKChsb2NhbGUgPSBzZXRsb2NhbGUoIGNhdCwgTlVMTCApKSkgXAogICAgeyBcCiAgICAgICAgc3RyY3B5bkF0b1coIGJ1ZmZlciwgbG9jYWxlLCBzaXplb2YoYnVmZmVyKS9zaXplb2YoV0NIQVIpICk7IFwKICAgICAgICBpZiAoIXN0cmNtcFcoIGJ1ZmZlciwgY3R5cGVfYnVmZiApKSBsY2lkXyMjY2F0ID0gbGNpZF9MQ19DVFlQRTsgXAogICAgICAgIGVsc2UgeyBcCiAgICAgICAgICAgIHBhcnNlX2xvY2FsZV9uYW1lKCBidWZmZXIsICZsb2NhbGVfbmFtZSApOyAgXAogICAgICAgICAgICBsY2lkXyMjY2F0ID0gbG9jYWxlX25hbWUubGNpZDsgXAogICAgICAgICAgICBUUkFDRSggImdvdCBsY2lkICUwNHggKCVkIG1hdGNoZXMpIGZvciAiICNjYXQgIj0lc1xuIiwgICAgICAgIFwKICAgICAgICAgICAgICAgICAgIGxvY2FsZV9uYW1lLmxjaWQsIGxvY2FsZV9uYW1lLm1hdGNoZXMsIGRlYnVnc3RyX2EobG9jYWxlKSApOyBcCiAgICAgICAgfSBcCiAgICB9IHdoaWxlICgwKQoKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfQ09MTEFURSApOwogICAgR0VUX1VOSVhfTE9DQUxFKCBMQ19NRVNTQUdFUyApOwogICAgR0VUX1VOSVhfTE9DQUxFKCBMQ19NT05FVEFSWSApOwogICAgR0VUX1VOSVhfTE9DQUxFKCBMQ19OVU1FUklDICk7CiAgICBHRVRfVU5JWF9MT0NBTEUoIExDX1RJTUUgKTsKI2lmZGVmIExDX1BBUEVSCiAgICBHRVRfVU5JWF9MT0NBTEUoIExDX1BBUEVSICk7CiNlbmRpZgojaWZkZWYgTENfTUVBU1VSRU1FTlQKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfTUVBU1VSRU1FTlQgKTsKI2VuZGlmCiNpZmRlZiBMQ19URUxFUEhPTkUKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfVEVMRVBIT05FICk7CiNlbmRpZgoKI3VuZGVmIEdFVF9VTklYX0xPQ0FMRQoKICAgIHJldHVybiB1bml4X2NwOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRVc2VyRGVmYXVsdExhbmdJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxhbmd1YWdlIElkIGZvciB0aGUgY3VycmVudCB1c2VyLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBsYW5ndWFnZSBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICovCkxBTkdJRCBXSU5BUEkgR2V0VXNlckRlZmF1bHRMYW5nSUQodm9pZCkKewogICAgcmV0dXJuIExBTkdJREZST01MQ0lEKEdldFVzZXJEZWZhdWx0TENJRCgpKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0U3lzdGVtRGVmYXVsdExhbmdJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxhbmd1YWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBsYW5ndWFnZSBmb3IgdGhlIHN5c3RlbS4KICovCkxBTkdJRCBXSU5BUEkgR2V0U3lzdGVtRGVmYXVsdExhbmdJRCh2b2lkKQp7CiAgICByZXR1cm4gTEFOR0lERlJPTUxDSUQoR2V0U3lzdGVtRGVmYXVsdExDSUQoKSk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUdldFVzZXJEZWZhdWx0TENJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxvY2FsZSBJZCBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQ0lEIG9mIHRoZSBkZWZhdWx0IGxvY2FsZSBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICovCkxDSUQgV0lOQVBJIEdldFVzZXJEZWZhdWx0TENJRCh2b2lkKQp7CiAgICBMQ0lEIGxjaWQ7CiAgICBOdFF1ZXJ5RGVmYXVsdExvY2FsZSggVFJVRSwgJmxjaWQgKTsKICAgIHJldHVybiBsY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRTeXN0ZW1EZWZhdWx0TENJRCAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IGxvY2FsZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQ0lEIG9mIHRoZSBkZWZhdWx0IGxvY2FsZSBmb3IgdGhlIHN5c3RlbS4KICovCkxDSUQgV0lOQVBJIEdldFN5c3RlbURlZmF1bHRMQ0lEKHZvaWQpCnsKICAgIExDSUQgbGNpZDsKICAgIE50UXVlcnlEZWZhdWx0TG9jYWxlKCBGQUxTRSwgJmxjaWQgKTsKICAgIHJldHVybiBsY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRVc2VyRGVmYXVsdFVJTGFuZ3VhZ2UgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCB1c2VyIGludGVyZmFjZSBsYW5ndWFnZSBJZCBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQU5HSUQgb2YgdGhlIGRlZmF1bHQgVUkgbGFuZ3VhZ2UgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqLwpMQU5HSUQgV0lOQVBJIEdldFVzZXJEZWZhdWx0VUlMYW5ndWFnZSh2b2lkKQp7CiAgICBMQU5HSUQgbGFuZzsKICAgIE50UXVlcnlEZWZhdWx0VUlMYW5ndWFnZSggJmxhbmcgKTsKICAgIHJldHVybiBsYW5nOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRTeXN0ZW1EZWZhdWx0VUlMYW5ndWFnZSAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBkZWZhdWx0IHVzZXIgaW50ZXJmYWNlIGxhbmd1YWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBjdXJyZW50IExBTkdJRCBvZiB0aGUgZGVmYXVsdCBVSSBsYW5ndWFnZSBmb3IgdGhlIHN5c3RlbS4gVGhpcyBpcwogKiAgdHlwaWNhbGx5IHRoZSBzYW1lIGxhbmd1YWdlIHVzZWQgZHVyaW5nIHRoZSBpbnN0YWxsYXRpb24gcHJvY2Vzcy4KICovCkxBTkdJRCBXSU5BUEkgR2V0U3lzdGVtRGVmYXVsdFVJTGFuZ3VhZ2Uodm9pZCkKewogICAgTEFOR0lEIGxhbmc7CiAgICBOdFF1ZXJ5SW5zdGFsbFVJTGFuZ3VhZ2UoICZsYW5nICk7CiAgICByZXR1cm4gbGFuZzsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgTG9jYWxlTmFtZVRvTENJRCAgKEtFUk5FTDMyLkApCiAqLwpMQ0lEIFdJTkFQSSBMb2NhbGVOYW1lVG9MQ0lEKCBMUENXU1RSIG5hbWUsIERXT1JEIGZsYWdzICkKewogICAgc3RydWN0IGxvY2FsZV9uYW1lIGxvY2FsZV9uYW1lOwoKICAgIGlmIChmbGFncykgRklYTUUoICJ1bnN1cHBvcnRlZCBmbGFncyAleFxuIiwgZmxhZ3MgKTsKCiAgICBwYXJzZV9sb2NhbGVfbmFtZSggbmFtZSwgJmxvY2FsZV9uYW1lICk7CgogICAgVFJBQ0UoICJmb3VuZCBsY2lkICV4IGZvciAlcywgbWF0Y2hlcyAlZFxuIiwKICAgICAgICAgICBsb2NhbGVfbmFtZS5sY2lkLCBkZWJ1Z3N0cl93KG5hbWUpLCBsb2NhbGVfbmFtZS5tYXRjaGVzICk7CgogICAgaWYgKCFsb2NhbGVfbmFtZS5tYXRjaGVzKQogICAgICAgIFdBUk4oICJsb2NhbGUgJXMgbm90IHJlY29nbml6ZWQsIGRlZmF1bHRpbmcgdG8gRW5nbGlzaFxuIiwgZGVidWdzdHJfdyhuYW1lKSApOwogICAgZWxzZSBpZiAobG9jYWxlX25hbWUubWF0Y2hlcyA9PSAxKQogICAgICAgIFdBUk4oICJsb2NhbGUgJXMgbm90IHJlY29nbml6ZWQsIGRlZmF1bHRpbmcgdG8gJXNcbiIsCiAgICAgICAgICAgICAgZGVidWdzdHJfdyhuYW1lKSwgZGVidWdzdHJfdyhsb2NhbGVfbmFtZS5sYW5nKSApOwoKICAgIHJldHVybiBsb2NhbGVfbmFtZS5sY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBMQ0lEVG9Mb2NhbGVOYW1lICAoS0VSTkVMMzIuQCkKICovCklOVCBXSU5BUEkgTENJRFRvTG9jYWxlTmFtZSggTENJRCBsY2lkLCBMUFdTVFIgbmFtZSwgSU5UIGNvdW50LCBEV09SRCBmbGFncyApCnsKICAgIGlmIChmbGFncykgRklYTUUoICJ1bnN1cHBvcnRlZCBmbGFncyAleFxuIiwgZmxhZ3MgKTsKCiAgICByZXR1cm4gR2V0TG9jYWxlSW5mb1coIGxjaWQsIExPQ0FMRV9TTkFNRSB8IExPQ0FMRV9OT1VTRVJPVkVSUklERSwgbmFtZSwgY291bnQgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWdldF9sb2NhbGVfdmFsdWVfbmFtZQogKgogKiBHZXRzIHRoZSByZWdpc3RyeSB2YWx1ZSBuYW1lIGZvciBhIGdpdmVuIGxjdHlwZS4KICovCnN0YXRpYyBjb25zdCBXQ0hBUiAqZ2V0X2xvY2FsZV92YWx1ZV9uYW1lKCBEV09SRCBsY3R5cGUgKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUNhbGVuZGFyVHlwZVdbXSA9IHsnaScsJ0MnLCdhJywnbCcsJ2UnLCduJywnZCcsJ2EnLCdyJywnVCcsJ3knLCdwJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUNvdW50cnlXW10gPSB7J2knLCdDJywnbycsJ3UnLCduJywndCcsJ3InLCd5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpQ3VyckRpZ2l0c1dbXSA9IHsnaScsJ0MnLCd1JywncicsJ3InLCdEJywnaScsJ2cnLCdpJywndCcsJ3MnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlDdXJyZW5jeVdbXSA9IHsnaScsJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ2MnLCd5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRGF0ZVdbXSA9IHsnaScsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlEaWdpdHNXW10gPSB7J2knLCdEJywnaScsJ2cnLCdpJywndCcsJ3MnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlGaXJzdERheU9mV2Vla1dbXSA9IHsnaScsJ0YnLCdpJywncicsJ3MnLCd0JywnRCcsJ2EnLCd5JywnTycsJ2YnLCdXJywnZScsJ2UnLCdrJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRmlyc3RXZWVrT2ZZZWFyV1tdID0geydpJywnRicsJ2knLCdyJywncycsJ3QnLCdXJywnZScsJ2UnLCdrJywnTycsJ2YnLCdZJywnZScsJ2EnLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpTERhdGVXW10gPSB7J2knLCdMJywnRCcsJ2EnLCd0JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUxaZXJvV1tdID0geydpJywnTCcsJ1onLCdlJywncicsJ28nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlNZWFzdXJlV1tdID0geydpJywnTScsJ2UnLCdhJywncycsJ3UnLCdyJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaU5lZ0N1cnJXW10gPSB7J2knLCdOJywnZScsJ2cnLCdDJywndScsJ3InLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpTmVnTnVtYmVyV1tdID0geydpJywnTicsJ2UnLCdnJywnTicsJ3UnLCdtJywnYicsJ2UnLCdyJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpUGFwZXJTaXplV1tdID0geydpJywnUCcsJ2EnLCdwJywnZScsJ3InLCdTJywnaScsJ3onLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpVExaZXJvV1tdID0geydpJywnVCcsJ0wnLCdaJywnZScsJ3InLCdvJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpVGltZVByZWZpeFdbXSA9IHsnaScsJ1QnLCdpJywnbScsJ2UnLCdQJywncicsJ2UnLCdmJywnaScsJ3gnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlUaW1lV1tdID0geydpJywnVCcsJ2knLCdtJywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgczExNTlXW10gPSB7J3MnLCcxJywnMScsJzUnLCc5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzMjM1OVdbXSA9IHsncycsJzInLCczJywnNScsJzknLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNDb3VudHJ5V1tdID0geydzJywnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0N1cnJlbmN5V1tdID0geydzJywnQycsJ3UnLCdyJywncicsJ2UnLCduJywnYycsJ3knLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNEYXRlV1tdID0geydzJywnRCcsJ2EnLCd0JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0RlY2ltYWxXW10gPSB7J3MnLCdEJywnZScsJ2MnLCdpJywnbScsJ2EnLCdsJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzR3JvdXBpbmdXW10gPSB7J3MnLCdHJywncicsJ28nLCd1JywncCcsJ2knLCduJywnZycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0xhbmd1YWdlV1tdID0geydzJywnTCcsJ2EnLCduJywnZycsJ3UnLCdhJywnZycsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNMaXN0V1tdID0geydzJywnTCcsJ2knLCdzJywndCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0xvbmdEYXRlV1tdID0geydzJywnTCcsJ28nLCduJywnZycsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNNb25EZWNpbWFsU2VwV1tdID0geydzJywnTScsJ28nLCduJywnRCcsJ2UnLCdjJywnaScsJ20nLCdhJywnbCcsJ1MnLCdlJywncCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc01vbkdyb3VwaW5nV1tdID0geydzJywnTScsJ28nLCduJywnRycsJ3InLCdvJywndScsJ3AnLCdpJywnbicsJ2cnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNNb25UaG91c2FuZFNlcFdbXSA9IHsncycsJ00nLCdvJywnbicsJ1QnLCdoJywnbycsJ3UnLCdzJywnYScsJ24nLCdkJywnUycsJ2UnLCdwJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTmF0aXZlRGlnaXRzV1tdID0geydzJywnTicsJ2EnLCd0JywnaScsJ3YnLCdlJywnRCcsJ2knLCdnJywnaScsJ3QnLCdzJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTmVnYXRpdmVTaWduV1tdID0geydzJywnTicsJ2UnLCdnJywnYScsJ3QnLCdpJywndicsJ2UnLCdTJywnaScsJ2cnLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzUG9zaXRpdmVTaWduV1tdID0geydzJywnUCcsJ28nLCdzJywnaScsJ3QnLCdpJywndicsJ2UnLCdTJywnaScsJ2cnLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzU2hvcnREYXRlV1tdID0geydzJywnUycsJ2gnLCdvJywncicsJ3QnLCdEJywnYScsJ3QnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzVGhvdXNhbmRXW10gPSB7J3MnLCdUJywnaCcsJ28nLCd1JywncycsJ2EnLCduJywnZCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1RpbWVGb3JtYXRXW10gPSB7J3MnLCdUJywnaScsJ20nLCdlJywnRicsJ28nLCdyJywnbScsJ2EnLCd0JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzVGltZVdbXSA9IHsncycsJ1QnLCdpJywnbScsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNZZWFyTW9udGhXW10gPSB7J3MnLCdZJywnZScsJ2EnLCdyJywnTScsJ28nLCduJywndCcsJ2gnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIE51bVNoYXBlV1tdID0geydOJywndScsJ20nLCdzJywnaCcsJ2EnLCdwJywnZScsMH07CgogICAgc3dpdGNoIChsY3R5cGUpCiAgICB7CiAgICAvKiBUaGVzZSB2YWx1ZXMgYXJlIHVzZWQgYnkgU2V0TG9jYWxlSW5mbyBhbmQgR2V0TG9jYWxlSW5mbywgYW5kCiAgICAgKiB0aGUgdmFsdWVzIGFyZSBzdG9yZWQgaW4gdGhlIHJlZ2lzdHJ5LCBjb25maXJtZWQgdW5kZXIgV2luZG93cy4KICAgICAqLwogICAgY2FzZSBMT0NBTEVfSUNBTEVOREFSVFlQRTogICAgcmV0dXJuIGlDYWxlbmRhclR5cGVXOwogICAgY2FzZSBMT0NBTEVfSUNVUlJESUdJVFM6ICAgICAgcmV0dXJuIGlDdXJyRGlnaXRzVzsKICAgIGNhc2UgTE9DQUxFX0lDVVJSRU5DWTogICAgICAgIHJldHVybiBpQ3VycmVuY3lXOwogICAgY2FzZSBMT0NBTEVfSURJR0lUUzogICAgICAgICAgcmV0dXJuIGlEaWdpdHNXOwogICAgY2FzZSBMT0NBTEVfSUZJUlNUREFZT0ZXRUVLOiAgcmV0dXJuIGlGaXJzdERheU9mV2Vla1c7CiAgICBjYXNlIExPQ0FMRV9JRklSU1RXRUVLT0ZZRUFSOiByZXR1cm4gaUZpcnN0V2Vla09mWWVhclc7CiAgICBjYXNlIExPQ0FMRV9JTFpFUk86ICAgICAgICAgICByZXR1cm4gaUxaZXJvVzsKICAgIGNhc2UgTE9DQUxFX0lNRUFTVVJFOiAgICAgICAgIHJldHVybiBpTWVhc3VyZVc7CiAgICBjYXNlIExPQ0FMRV9JTkVHQ1VSUjogICAgICAgICByZXR1cm4gaU5lZ0N1cnJXOwogICAgY2FzZSBMT0NBTEVfSU5FR05VTUJFUjogICAgICAgcmV0dXJuIGlOZWdOdW1iZXJXOwogICAgY2FzZSBMT0NBTEVfSVBBUEVSU0laRTogICAgICAgcmV0dXJuIGlQYXBlclNpemVXOwogICAgY2FzZSBMT0NBTEVfSVRJTUU6ICAgICAgICAgICAgcmV0dXJuIGlUaW1lVzsKICAgIGNhc2UgTE9DQUxFX1MxMTU5OiAgICAgICAgICAgIHJldHVybiBzMTE1OVc7CiAgICBjYXNlIExPQ0FMRV9TMjM1OTogICAgICAgICAgICByZXR1cm4gczIzNTlXOwogICAgY2FzZSBMT0NBTEVfU0NVUlJFTkNZOiAgICAgICAgcmV0dXJuIHNDdXJyZW5jeVc7CiAgICBjYXNlIExPQ0FMRV9TREFURTogICAgICAgICAgICByZXR1cm4gc0RhdGVXOwogICAgY2FzZSBMT0NBTEVfU0RFQ0lNQUw6ICAgICAgICAgcmV0dXJuIHNEZWNpbWFsVzsKICAgIGNhc2UgTE9DQUxFX1NHUk9VUElORzogICAgICAgIHJldHVybiBzR3JvdXBpbmdXOwogICAgY2FzZSBMT0NBTEVfU0xJU1Q6ICAgICAgICAgICAgcmV0dXJuIHNMaXN0VzsKICAgIGNhc2UgTE9DQUxFX1NMT05HREFURTogICAgICAgIHJldHVybiBzTG9uZ0RhdGVXOwogICAgY2FzZSBMT0NBTEVfU01PTkRFQ0lNQUxTRVA6ICAgcmV0dXJuIHNNb25EZWNpbWFsU2VwVzsKICAgIGNhc2UgTE9DQUxFX1NNT05HUk9VUElORzogICAgIHJldHVybiBzTW9uR3JvdXBpbmdXOwogICAgY2FzZSBMT0NBTEVfU01PTlRIT1VTQU5EU0VQOiAgcmV0dXJuIHNNb25UaG91c2FuZFNlcFc7CiAgICBjYXNlIExPQ0FMRV9TTkVHQVRJVkVTSUdOOiAgICByZXR1cm4gc05lZ2F0aXZlU2lnblc7CiAgICBjYXNlIExPQ0FMRV9TUE9TSVRJVkVTSUdOOiAgICByZXR1cm4gc1Bvc2l0aXZlU2lnblc7CiAgICBjYXNlIExPQ0FMRV9TU0hPUlREQVRFOiAgICAgICByZXR1cm4gc1Nob3J0RGF0ZVc7CiAgICBjYXNlIExPQ0FMRV9TVEhPVVNBTkQ6ICAgICAgICByZXR1cm4gc1Rob3VzYW5kVzsKICAgIGNhc2UgTE9DQUxFX1NUSU1FOiAgICAgICAgICAgIHJldHVybiBzVGltZVc7CiAgICBjYXNlIExPQ0FMRV9TVElNRUZPUk1BVDogICAgICByZXR1cm4gc1RpbWVGb3JtYXRXOwogICAgY2FzZSBMT0NBTEVfU1lFQVJNT05USDogICAgICAgcmV0dXJuIHNZZWFyTW9udGhXOwoKICAgIC8qIFRoZSBmb2xsb3dpbmcgYXJlIG5vdCBsaXN0ZWQgdW5kZXIgTVNETiBhcyBzdXBwb3J0ZWQsCiAgICAgKiBidXQgc2VlbSB0byBiZSB1c2VkIGFuZCBhbHNvIHN0b3JlZCBpbiB0aGUgcmVnaXN0cnkuCiAgICAgKi8KICAgIGNhc2UgTE9DQUxFX0lDT1VOVFJZOiAgICAgICAgIHJldHVybiBpQ291bnRyeVc7CiAgICBjYXNlIExPQ0FMRV9JREFURTogICAgICAgICAgICByZXR1cm4gaURhdGVXOwogICAgY2FzZSBMT0NBTEVfSUxEQVRFOiAgICAgICAgICAgcmV0dXJuIGlMRGF0ZVc7CiAgICBjYXNlIExPQ0FMRV9JVExaRVJPOiAgICAgICAgICByZXR1cm4gaVRMWmVyb1c7CiAgICBjYXNlIExPQ0FMRV9TQ09VTlRSWTogICAgICAgICByZXR1cm4gc0NvdW50cnlXOwogICAgY2FzZSBMT0NBTEVfU0xBTkdVQUdFOiAgICAgICAgcmV0dXJuIHNMYW5ndWFnZVc7CgogICAgLyogVGhlIGZvbGxvd2luZyBhcmUgdXNlZCBpbiBYUCBhbmQgbGF0ZXIgKi8KICAgIGNhc2UgTE9DQUxFX0lESUdJVFNVQlNUSVRVVElPTjogcmV0dXJuIE51bVNoYXBlVzsKICAgIGNhc2UgTE9DQUxFX1NOQVRJVkVESUdJVFM6ICAgICAgcmV0dXJuIHNOYXRpdmVEaWdpdHNXOwogICAgY2FzZSBMT0NBTEVfSVRJTUVNQVJLUE9TTjogICAgICByZXR1cm4gaVRpbWVQcmVmaXhXOwogICAgfQogICAgcmV0dXJuIE5VTEw7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8KICoKICogUmV0cmlldmUgdXNlci1tb2RpZmllZCBsb2NhbGUgaW5mbyBmcm9tIHRoZSByZWdpc3RyeS4KICogUmV0dXJuIGxlbmd0aCwgMCBvbiBlcnJvciwgLTEgaWYgbm90IGZvdW5kLgogKi8Kc3RhdGljIElOVCBnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8oIExQQ1dTVFIgdmFsdWUsIExQV1NUUiBidWZmZXIsIElOVCBsZW4gKQp7CiAgICBEV09SRCBzaXplOwogICAgSU5UIHJldDsKICAgIEhBTkRMRSBoa2V5OwogICAgTlRTVEFUVVMgc3RhdHVzOwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVc7CiAgICBLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqaW5mbzsKICAgIHN0YXRpYyBjb25zdCBpbnQgaW5mb19zaXplID0gRklFTERfT0ZGU0VUKEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OLCBEYXRhKTsKCiAgICBpZiAoIShoa2V5ID0gY3JlYXRlX3JlZ2lzdHJ5X2tleSgpKSkgcmV0dXJuIC0xOwoKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmbmFtZVcsIHZhbHVlICk7CiAgICBzaXplID0gaW5mb19zaXplICsgbGVuICogc2l6ZW9mKFdDSEFSKTsKCiAgICBpZiAoIShpbmZvID0gSGVhcEFsbG9jKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBzaXplICkpKQogICAgewogICAgICAgIE50Q2xvc2UoIGhrZXkgKTsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgc3RhdHVzID0gTnRRdWVyeVZhbHVlS2V5KCBoa2V5LCAmbmFtZVcsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLCBpbmZvLCBzaXplLCAmc2l6ZSApOwoKICAgIGlmICghc3RhdHVzKQogICAgewogICAgICAgIHJldCA9IChzaXplIC0gaW5mb19zaXplKSAvIHNpemVvZihXQ0hBUik7CiAgICAgICAgLyogYXBwZW5kIHRlcm1pbmF0aW5nIG51bGwgaWYgbmVlZGVkICovCiAgICAgICAgaWYgKCFyZXQgfHwgKChXQ0hBUiAqKWluZm8tPkRhdGEpW3JldC0xXSkKICAgICAgICB7CiAgICAgICAgICAgIGlmIChyZXQgPCBsZW4gfHwgIWJ1ZmZlcikgcmV0Kys7CiAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSICk7CiAgICAgICAgICAgICAgICByZXQgPSAwOwogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGlmIChyZXQgJiYgYnVmZmVyKQogICAgICAgIHsKICAgICAgICAgICAgbWVtY3B5KCBidWZmZXIsIGluZm8tPkRhdGEsIChyZXQtMSkgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgICAgIGJ1ZmZlcltyZXQtMV0gPSAwOwogICAgICAgIH0KICAgIH0KICAgIGVsc2UgaWYgKHN0YXR1cyA9PSBTVEFUVVNfQlVGRkVSX09WRVJGTE9XICYmICFidWZmZXIpCiAgICB7CiAgICAgICAgcmV0ID0gKHNpemUgLSBpbmZvX3NpemUpIC8gc2l6ZW9mKFdDSEFSKSArIDE7CiAgICB9CiAgICBlbHNlIGlmIChzdGF0dXMgPT0gU1RBVFVTX09CSkVDVF9OQU1FX05PVF9GT1VORCkKICAgIHsKICAgICAgICByZXQgPSAtMTsKICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIFJ0bE50U3RhdHVzVG9Eb3NFcnJvcihzdGF0dXMpICk7CiAgICAgICAgcmV0ID0gMDsKICAgIH0KICAgIE50Q2xvc2UoIGhrZXkgKTsKICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBpbmZvICk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0TG9jYWxlSW5mb0EgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBpbmZvcm1hdGlvbiBhYm91dCBhbiBhc3BlY3Qgb2YgYSBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUKICogIGxjdHlwZSBbSV0gTENUWVBFXyBmbGFncyBmcm9tICJ3aW5ubHMuaCIKICogIGJ1ZmZlciBbT10gRGVzdGluYXRpb24gZm9yIHRoZSBpbmZvcm1hdGlvbgogKiAgbGVuICAgIFtJXSBMZW5ndGggb2YgYnVmZmVyIGluIGNoYXJhY3RlcnMKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVGhlIHNpemUgb2YgdGhlIGRhdGEgcmVxdWVzdGVkLiBJZiBidWZmZXIgaXMgbm9uLU5VTEwsIGl0IGlzIGZpbGxlZAogKiAgICAgICAgICAgd2l0aCB0aGUgaW5mb3JtYXRpb24uCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICogTk9URVMKICogIC0gTE9DQUxFX05FVVRSQUwgaXMgZXF1YWwgdG8gTE9DQUxFX1NZU1RFTV9ERUZBVUxUCiAqICAtIFRoZSBzdHJpbmcgcmV0dXJuZWQgaXMgTlVMIHRlcm1pbmF0ZWQsIGV4Y2VwdCBmb3IgTE9DQUxFX0ZPTlRTSUdOQVRVUkUsCiAqICAgIHdoaWNoIGlzIGEgYml0IHN0cmluZy4KICovCklOVCBXSU5BUEkgR2V0TG9jYWxlSW5mb0EoIExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBTVFIgYnVmZmVyLCBJTlQgbGVuICkKewogICAgV0NIQVIgKmJ1ZmZlclc7CiAgICBJTlQgbGVuVywgcmV0OwoKICAgIFRSQUNFKCAiKGxjaWQ9MHgleCxsY3R5cGU9MHgleCwlcCwlZClcbiIsIGxjaWQsIGxjdHlwZSwgYnVmZmVyLCBsZW4gKTsKCiAgICBpZiAobGVuIDwgMCB8fCAobGVuICYmICFidWZmZXIpKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIGlmICghbGVuKSBidWZmZXIgPSBOVUxMOwoKICAgIGlmICghKGxlblcgPSBHZXRMb2NhbGVJbmZvVyggbGNpZCwgbGN0eXBlLCBOVUxMLCAwICkpKSByZXR1cm4gMDsKCiAgICBpZiAoIShidWZmZXJXID0gSGVhcEFsbG9jKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBsZW5XICogc2l6ZW9mKFdDSEFSKSApKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CiAgICBpZiAoKHJldCA9IEdldExvY2FsZUluZm9XKCBsY2lkLCBsY3R5cGUsIGJ1ZmZlclcsIGxlblcgKSkpCiAgICB7CiAgICAgICAgaWYgKChsY3R5cGUgJiBMT0NBTEVfUkVUVVJOX05VTUJFUikgfHwKICAgICAgICAgICAgKChsY3R5cGUgJiB+TE9DQUxFX0xPQ0FMRUlORk9GTEFHU01BU0spID09IExPQ0FMRV9GT05UU0lHTkFUVVJFKSkKICAgICAgICB7CiAgICAgICAgICAgIC8qIGl0J3Mgbm90IGFuIEFTQ0lJIHN0cmluZywganVzdCBieXRlcyAqLwogICAgICAgICAgICByZXQgKj0gc2l6ZW9mKFdDSEFSKTsKICAgICAgICAgICAgaWYgKGJ1ZmZlcikKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgaWYgKHJldCA8PSBsZW4pIG1lbWNweSggYnVmZmVyLCBidWZmZXJXLCByZXQgKTsKICAgICAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsKICAgICAgICAgICAgICAgICAgICByZXQgPSAwOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIFVJTlQgY29kZXBhZ2UgPSBDUF9BQ1A7CiAgICAgICAgICAgIGlmICghKGxjdHlwZSAmIExPQ0FMRV9VU0VfQ1BfQUNQKSkgY29kZXBhZ2UgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwogICAgICAgICAgICByZXQgPSBXaWRlQ2hhclRvTXVsdGlCeXRlKCBjb2RlcGFnZSwgMCwgYnVmZmVyVywgcmV0LCBidWZmZXIsIGxlbiwgTlVMTCwgTlVMTCApOwogICAgICAgIH0KICAgIH0KICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBidWZmZXJXICk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0TG9jYWxlSW5mb1cgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBHZXRMb2NhbGVJbmZvQS4KICovCklOVCBXSU5BUEkgR2V0TG9jYWxlSW5mb1coIExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBXU1RSIGJ1ZmZlciwgSU5UIGxlbiApCnsKICAgIExBTkdJRCBsYW5nX2lkOwogICAgSFJTUkMgaHJzcmM7CiAgICBIR0xPQkFMIGhtZW07CiAgICBJTlQgcmV0OwogICAgVUlOVCBsY2ZsYWdzOwogICAgY29uc3QgV0NIQVIgKnA7CiAgICB1bnNpZ25lZCBpbnQgaTsKCiAgICBpZiAobGVuIDwgMCB8fCAobGVuICYmICFidWZmZXIpKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIGlmICghbGVuKSBidWZmZXIgPSBOVUxMOwoKICAgIGxjaWQgPSBjb252ZXJ0X2RlZmF1bHRfbGNpZCggbGNpZCwgbGN0eXBlICk7CgogICAgbGNmbGFncyA9IGxjdHlwZSAmIExPQ0FMRV9MT0NBTEVJTkZPRkxBR1NNQVNLOwogICAgbGN0eXBlICY9IDB4ZmZmZjsKCiAgICBUUkFDRSggIihsY2lkPTB4JXgsbGN0eXBlPTB4JXgsJXAsJWQpXG4iLCBsY2lkLCBsY3R5cGUsIGJ1ZmZlciwgbGVuICk7CgogICAgLyogZmlyc3QgY2hlY2sgZm9yIG92ZXJyaWRlcyBpbiB0aGUgcmVnaXN0cnkgKi8KCiAgICBpZiAoIShsY2ZsYWdzICYgTE9DQUxFX05PVVNFUk9WRVJSSURFKSAmJgogICAgICAgIGxjaWQgPT0gY29udmVydF9kZWZhdWx0X2xjaWQoIExPQ0FMRV9VU0VSX0RFRkFVTFQsIGxjdHlwZSApKQogICAgewogICAgICAgIGNvbnN0IFdDSEFSICp2YWx1ZSA9IGdldF9sb2NhbGVfdmFsdWVfbmFtZShsY3R5cGUpOwoKICAgICAgICBpZiAodmFsdWUpCiAgICAgICAgewogICAgICAgICAgICBpZiAobGNmbGFncyAmIExPQ0FMRV9SRVRVUk5fTlVNQkVSKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBXQ0hBUiB0bXBbMTZdOwogICAgICAgICAgICAgICAgcmV0ID0gZ2V0X3JlZ2lzdHJ5X2xvY2FsZV9pbmZvKCB2YWx1ZSwgdG1wLCBzaXplb2YodG1wKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgICAgICAgICBpZiAocmV0ID4gMCkKICAgICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgICAgICBXQ0hBUiAqZW5kOwogICAgICAgICAgICAgICAgICAgIFVJTlQgbnVtYmVyID0gc3RydG9sVyggdG1wLCAmZW5kLCAxMCApOwogICAgICAgICAgICAgICAgICAgIGlmICgqZW5kKSAgLyogaW52YWxpZCBudW1iZXIgKi8KICAgICAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9GTEFHUyApOwogICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgcmV0ID0gc2l6ZW9mKFVJTlQpL3NpemVvZihXQ0hBUik7CiAgICAgICAgICAgICAgICAgICAgaWYgKCFidWZmZXIpIHJldHVybiByZXQ7CiAgICAgICAgICAgICAgICAgICAgaWYgKHJldCA+IGxlbikKICAgICAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUiApOwogICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgbWVtY3B5KCBidWZmZXIsICZudW1iZXIsIHNpemVvZihudW1iZXIpICk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgZWxzZSByZXQgPSBnZXRfcmVnaXN0cnlfbG9jYWxlX2luZm8oIHZhbHVlLCBidWZmZXIsIGxlbiApOwoKICAgICAgICAgICAgaWYgKHJldCAhPSAtMSkgcmV0dXJuIHJldDsKICAgICAgICB9CiAgICB9CgogICAgLyogbm93IGxvYWQgaXQgZnJvbSBrZXJuZWwgcmVzb3VyY2VzICovCgogICAgbGFuZ19pZCA9IExBTkdJREZST01MQ0lEKCBsY2lkICk7CgogICAgLyogcmVwbGFjZSBTVUJMQU5HX05FVVRSQUwgYnkgU1VCTEFOR19ERUZBVUxUICovCiAgICBpZiAoU1VCTEFOR0lEKGxhbmdfaWQpID09IFNVQkxBTkdfTkVVVFJBTCkKICAgICAgICBsYW5nX2lkID0gTUFLRUxBTkdJRChQUklNQVJZTEFOR0lEKGxhbmdfaWQpLCBTVUJMQU5HX0RFRkFVTFQpOwoKICAgIGlmICghKGhyc3JjID0gRmluZFJlc291cmNlRXhXKCBrZXJuZWwzMl9oYW5kbGUsIChMUFdTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVMb25nVG9QdHIoKGxjdHlwZSA+PiA0KSArIDEpLCBsYW5nX2lkICkpKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9GTEFHUyApOyAgLyogbm8gc3VjaCBsY3R5cGUgKi8KICAgICAgICByZXR1cm4gMDsKICAgIH0KICAgIGlmICghKGhtZW0gPSBMb2FkUmVzb3VyY2UoIGtlcm5lbDMyX2hhbmRsZSwgaHJzcmMgKSkpCiAgICAgICAgcmV0dXJuIDA7CgogICAgcCA9IExvY2tSZXNvdXJjZSggaG1lbSApOwogICAgZm9yIChpID0gMDsgaSA8IChsY3R5cGUgJiAweDBmKTsgaSsrKSBwICs9ICpwICsgMTsKCiAgICBpZiAobGNmbGFncyAmIExPQ0FMRV9SRVRVUk5fTlVNQkVSKSByZXQgPSBzaXplb2YoVUlOVCkvc2l6ZW9mKFdDSEFSKTsKICAgIGVsc2UgcmV0ID0gKGxjdHlwZSA9PSBMT0NBTEVfRk9OVFNJR05BVFVSRSkgPyAqcCA6ICpwICsgMTsKCiAgICBpZiAoIWJ1ZmZlcikgcmV0dXJuIHJldDsKCiAgICBpZiAocmV0ID4gbGVuKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChsY2ZsYWdzICYgTE9DQUxFX1JFVFVSTl9OVU1CRVIpCiAgICB7CiAgICAgICAgVUlOVCBudW1iZXI7CiAgICAgICAgV0NIQVIgKmVuZCwgKnRtcCA9IEhlYXBBbGxvYyggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgKCpwICsgMSkgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgaWYgKCF0bXApIHJldHVybiAwOwogICAgICAgIG1lbWNweSggdG1wLCBwICsgMSwgKnAgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgdG1wWypwXSA9IDA7CiAgICAgICAgbnVtYmVyID0gc3RydG9sVyggdG1wLCAmZW5kLCAxMCApOwogICAgICAgIGlmICghKmVuZCkKICAgICAgICAgICAgbWVtY3B5KCBidWZmZXIsICZudW1iZXIsIHNpemVvZihudW1iZXIpICk7CiAgICAgICAgZWxzZSAgLyogaW52YWxpZCBudW1iZXIgKi8KICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9GTEFHUyApOwogICAgICAgICAgICByZXQgPSAwOwogICAgICAgIH0KICAgICAgICBIZWFwRnJlZSggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgdG1wICk7CgogICAgICAgIFRSQUNFKCAiKGxjaWQ9MHgleCxsY3R5cGU9MHgleCwlcCwlZCkgcmV0dXJuaW5nIG51bWJlciAlZFxuIiwKICAgICAgICAgICAgICAgbGNpZCwgbGN0eXBlLCBidWZmZXIsIGxlbiwgbnVtYmVyICk7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgbWVtY3B5KCBidWZmZXIsIHAgKyAxLCAqcCAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICBpZiAobGN0eXBlICE9IExPQ0FMRV9GT05UU0lHTkFUVVJFKSBidWZmZXJbcmV0LTFdID0gMDsKCiAgICAgICAgVFJBQ0UoICIobGNpZD0weCV4LGxjdHlwZT0weCV4LCVwLCVkKSByZXR1cm5pbmcgJWQgJXNcbiIsCiAgICAgICAgICAgICAgIGxjaWQsIGxjdHlwZSwgYnVmZmVyLCBsZW4sIHJldCwgZGVidWdzdHJfdyhidWZmZXIpICk7CiAgICB9CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJU2V0TG9jYWxlSW5mb0EJW0tFUk5FTDMyLkBdCiAqCiAqIFNldCBpbmZvcm1hdGlvbiBhYm91dCBhbiBhc3BlY3Qgb2YgYSBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUKICogIGxjdHlwZSBbSV0gTENUWVBFXyBmbGFncyBmcm9tICJ3aW5ubHMuaCIKICogIGRhdGEgICBbSV0gSW5mb3JtYXRpb24gdG8gc2V0CiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIFRoZSBpbmZvcm1hdGlvbiBnaXZlbiB3aWxsIGJlIHJldHVybmVkIGJ5IEdldExvY2FsZUluZm9BKCkKICogICAgICAgICAgIHdoZW5ldmVyIGl0IGlzIGNhbGxlZCB3aXRob3V0IExPQ0FMRV9OT1VTRVJPVkVSUklERS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICogTk9URVMKICogIC0gVmFsdWVzIGFyZSBvbmx5IGJlIHNldCBmb3IgdGhlIGN1cnJlbnQgdXNlciBsb2NhbGU7IHRoZSBzeXN0ZW0gbG9jYWxlCiAqICBzZXR0aW5ncyBjYW5ub3QgYmUgY2hhbmdlZC4KICogIC0gQW55IHNldHRpbmdzIGNoYW5nZWQgYnkgdGhpcyBjYWxsIGFyZSBsb3N0IHdoZW4gdGhlIGxvY2FsZSBpcyBjaGFuZ2VkIGJ5CiAqICB0aGUgY29udHJvbCBwYW5lbCAoaW4gV2luZSwgdGhpcyBoYXBwZW5zIGV2ZXJ5IHRpbWUgeW91IGNoYW5nZSBMQU5HKS4KICogIC0gVGhlIG5hdGl2ZSBpbXBsZW1lbnRhdGlvbiBvZiB0aGlzIGZ1bmN0aW9uIGRvZXMgbm90IGNoZWNrIHRoYXQgbGNpZCBtYXRjaGVzCiAqICB0aGUgY3VycmVudCB1c2VyIGxvY2FsZSwgYW5kIHNpbXBseSBzZXRzIHRoZSBuZXcgdmFsdWVzLiBXaW5lIHdhcm5zIHlvdSBpbgogKiAgdGhpcyBjYXNlLCBidXQgYmVoYXZlcyB0aGUgc2FtZS4KICovCkJPT0wgV0lOQVBJIFNldExvY2FsZUluZm9BKExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBDU1RSIGRhdGEpCnsKICAgIFVJTlQgY29kZXBhZ2UgPSBDUF9BQ1A7CiAgICBXQ0hBUiAqc3RyVzsKICAgIERXT1JEIGxlbjsKICAgIEJPT0wgcmV0OwoKICAgIGlmICghKGxjdHlwZSAmIExPQ0FMRV9VU0VfQ1BfQUNQKSkgY29kZXBhZ2UgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwoKICAgIGlmICghZGF0YSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgbGVuID0gTXVsdGlCeXRlVG9XaWRlQ2hhciggY29kZXBhZ2UsIDAsIGRhdGEsIC0xLCBOVUxMLCAwICk7CiAgICBpZiAoIShzdHJXID0gSGVhcEFsbG9jKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBsZW4gKiBzaXplb2YoV0NIQVIpICkpKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkgKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CiAgICBNdWx0aUJ5dGVUb1dpZGVDaGFyKCBjb2RlcGFnZSwgMCwgZGF0YSwgLTEsIHN0clcsIGxlbiApOwogICAgcmV0ID0gU2V0TG9jYWxlSW5mb1coIGxjaWQsIGxjdHlwZSwgc3RyVyApOwogICAgSGVhcEZyZWUoIEdldFByb2Nlc3NIZWFwKCksIDAsIHN0clcgKTsKICAgIHJldHVybiByZXQ7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlTZXRMb2NhbGVJbmZvVwkoS0VSTkVMMzIuQCkKICoKICogU2VlIFNldExvY2FsZUluZm9BLgogKi8KQk9PTCBXSU5BUEkgU2V0TG9jYWxlSW5mb1coIExDSUQgbGNpZCwgTENUWVBFIGxjdHlwZSwgTFBDV1NUUiBkYXRhICkKewogICAgY29uc3QgV0NIQVIgKnZhbHVlOwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGludGxXW10gPSB7J2knLCduJywndCcsJ2wnLDAgfTsKICAgIFVOSUNPREVfU1RSSU5HIHZhbHVlVzsKICAgIE5UU1RBVFVTIHN0YXR1czsKICAgIEhBTkRMRSBoa2V5OwoKICAgIGxjdHlwZSAmPSAweGZmZmY7CiAgICB2YWx1ZSA9IGdldF9sb2NhbGVfdmFsdWVfbmFtZSggbGN0eXBlICk7CgogICAgaWYgKCFkYXRhIHx8ICF2YWx1ZSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmIChsY3R5cGUgPT0gTE9DQUxFX0lEQVRFIHx8IGxjdHlwZSA9PSBMT0NBTEVfSUxEQVRFKQogICAgewogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9GTEFHUyApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBUUkFDRSgic2V0dGluZyAleCAoJXMpIHRvICVzXG4iLCBsY3R5cGUsIGRlYnVnc3RyX3codmFsdWUpLCBkZWJ1Z3N0cl93KGRhdGEpICk7CgogICAgLyogRklYTUU6IHNob3VsZCBjaGVjayB0aGF0IGRhdGEgdG8gc2V0IGlzIHNhbmUgKi8KCiAgICAvKiBGSVhNRTogcHJvZmlsZSBmdW5jdGlvbnMgc2hvdWxkIG1hcCB0byByZWdpc3RyeSAqLwogICAgV3JpdGVQcm9maWxlU3RyaW5nVyggaW50bFcsIHZhbHVlLCBkYXRhICk7CgogICAgaWYgKCEoaGtleSA9IGNyZWF0ZV9yZWdpc3RyeV9rZXkoKSkpIHJldHVybiBGQUxTRTsKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmdmFsdWVXLCB2YWx1ZSApOwogICAgc3RhdHVzID0gTnRTZXRWYWx1ZUtleSggaGtleSwgJnZhbHVlVywgMCwgUkVHX1NaLCBkYXRhLCAoc3RybGVuVyhkYXRhKSsxKSpzaXplb2YoV0NIQVIpICk7CgogICAgaWYgKGxjdHlwZSA9PSBMT0NBTEVfU1NIT1JUREFURSB8fCBsY3R5cGUgPT0gTE9DQUxFX1NMT05HREFURSkKICAgIHsKICAgICAgLyogU2V0IEktdmFsdWUgZnJvbSBTIHZhbHVlICovCiAgICAgIFdDSEFSICpscEQsICpscE0sICpscFk7CiAgICAgIFdDSEFSIHN6QnVmZlsyXTsKCiAgICAgIGxwRCA9IHN0cnJjaHJXKGRhdGEsICdkJyk7CiAgICAgIGxwTSA9IHN0cnJjaHJXKGRhdGEsICdNJyk7CiAgICAgIGxwWSA9IHN0cnJjaHJXKGRhdGEsICd5Jyk7CgogICAgICBpZiAobHBEIDw9IGxwTSkKICAgICAgewogICAgICAgIHN6QnVmZlswXSA9ICcxJzsgLyogRC1NLVkgKi8KICAgICAgfQogICAgICBlbHNlCiAgICAgIHsKICAgICAgICBpZiAobHBZIDw9IGxwTSkKICAgICAgICAgIHN6QnVmZlswXSA9ICcyJzsgLyogWS1NLUQgKi8KICAgICAgICBlbHNlCiAgICAgICAgICBzekJ1ZmZbMF0gPSAnMCc7IC8qIE0tRC1ZICovCiAgICAgIH0KCiAgICAgIHN6QnVmZlsxXSA9ICdcMCc7CgogICAgICBpZiAobGN0eXBlID09IExPQ0FMRV9TU0hPUlREQVRFKQogICAgICAgIGxjdHlwZSA9IExPQ0FMRV9JREFURTsKICAgICAgZWxzZQogICAgICAgIGxjdHlwZSA9IExPQ0FMRV9JTERBVEU7CgogICAgICB2YWx1ZSA9IGdldF9sb2NhbGVfdmFsdWVfbmFtZSggbGN0eXBlICk7CgogICAgICBXcml0ZVByb2ZpbGVTdHJpbmdXKCBpbnRsVywgdmFsdWUsIHN6QnVmZiApOwoKICAgICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZ2YWx1ZVcsIHZhbHVlICk7CiAgICAgIHN0YXR1cyA9IE50U2V0VmFsdWVLZXkoIGhrZXksICZ2YWx1ZVcsIDAsIFJFR19TWiwgc3pCdWZmLCBzaXplb2Yoc3pCdWZmKSApOwogICAgfQoKICAgIE50Q2xvc2UoIGhrZXkgKTsKCiAgICBpZiAoc3RhdHVzKSBTZXRMYXN0RXJyb3IoIFJ0bE50U3RhdHVzVG9Eb3NFcnJvcihzdGF0dXMpICk7CiAgICByZXR1cm4gIXN0YXR1czsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEdldEFDUCAgIChLRVJORUwzMi5AKQogKgogKiBHZXQgdGhlIGN1cnJlbnQgQW5zaSBjb2RlIHBhZ2UgSWQgZm9yIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgICBUaGUgY3VycmVudCBBbnNpIGNvZGUgcGFnZSBpZGVudGlmaWVyIGZvciB0aGUgc3lzdGVtLgogKi8KVUlOVCBXSU5BUEkgR2V0QUNQKHZvaWQpCnsKICAgIGFzc2VydCggYW5zaV9jcHRhYmxlICk7CiAgICByZXR1cm4gYW5zaV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgICAgU2V0Q1BHbG9iYWwgICAoS0VSTkVMMzIuQCkKICoKICogU2V0IHRoZSBjdXJyZW50IEFuc2kgY29kZSBwYWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogICAgYWNwIFtJXSBjb2RlIHBhZ2UgSUQgdG8gYmUgdGhlIG5ldyBBQ1AuCiAqCiAqIFJFVFVSTlMKICogICAgVGhlIHByZXZpb3VzIEFDUC4KICovClVJTlQgV0lOQVBJIFNldENQR2xvYmFsKCBVSU5UIGFjcCApCnsKICAgIFVJTlQgcmV0ID0gR2V0QUNQKCk7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICpuZXdfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBhY3AgKTsKCiAgICBpZiAobmV3X2NwdGFibGUpIGFuc2lfY3B0YWJsZSA9IG5ld19jcHRhYmxlOwogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgICAgR2V0T0VNQ1AgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBjdXJyZW50IE9FTSBjb2RlIHBhZ2UgSWQgZm9yIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgICBUaGUgY3VycmVudCBPRU0gY29kZSBwYWdlIGlkZW50aWZpZXIgZm9yIHRoZSBzeXN0ZW0uCiAqLwpVSU5UIFdJTkFQSSBHZXRPRU1DUCh2b2lkKQp7CiAgICBhc3NlcnQoIG9lbV9jcHRhYmxlICk7CiAgICByZXR1cm4gb2VtX2NwdGFibGUtPmluZm8uY29kZXBhZ2U7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIElzVmFsaWRDb2RlUGFnZSAgIChLRVJORUwzMi5AKQogKgogKiBEZXRlcm1pbmUgaWYgYSBnaXZlbiBjb2RlIHBhZ2UgaWRlbnRpZmllciBpcyB2YWxpZC4KICoKICogUEFSQU1TCiAqICBjb2RlcGFnZSBbSV0gQ29kZSBwYWdlIElkIHRvIHZlcmlmeS4KICoKICogUkVUVVJOUwogKiAgVFJVRSwgSWYgY29kZXBhZ2UgaXMgdmFsaWQgYW5kIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLAogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNWYWxpZENvZGVQYWdlKCBVSU5UIGNvZGVwYWdlICkKewogICAgc3dpdGNoKGNvZGVwYWdlKSB7CiAgICBjYXNlIENQX1VURjc6CiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgcmV0dXJuIFRSVUU7CiAgICBkZWZhdWx0OgogICAgICAgIHJldHVybiB3aW5lX2NwX2dldF90YWJsZSggY29kZXBhZ2UgKSAhPSBOVUxMOwogICAgfQp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc0RCQ1NMZWFkQnl0ZUV4ICAgKEtFUk5FTDMyLkApCiAqCiAqIERldGVybWluZSBpZiBhIGNoYXJhY3RlciBpcyBhIGxlYWQgYnl0ZSBpbiBhIGdpdmVuIGNvZGUgcGFnZS4KICoKICogUEFSQU1TCiAqICBjb2RlcGFnZSBbSV0gQ29kZSBwYWdlIGZvciB0aGUgdGVzdC4KICogIHRlc3RjaGFyIFtJXSBDaGFyYWN0ZXIgdG8gdGVzdAogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBpZiB0ZXN0Y2hhciBpcyBhIGxlYWQgYnl0ZSBpbiBjb2RlcGFnZSwKICogIEZBTFNFIG90aGVyd2lzZS4KICovCkJPT0wgV0lOQVBJIElzREJDU0xlYWRCeXRlRXgoIFVJTlQgY29kZXBhZ2UsIEJZVEUgdGVzdGNoYXIgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZSA9IGdldF9jb2RlcGFnZV90YWJsZSggY29kZXBhZ2UgKTsKICAgIHJldHVybiB0YWJsZSAmJiB3aW5lX2lzX2RiY3NfbGVhZGJ5dGUoIHRhYmxlLCB0ZXN0Y2hhciApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc0RCQ1NMZWFkQnl0ZSAgIChLRVJORUwzMi5AKQogKiAgICAgICAgICAgSXNEQkNTTGVhZEJ5dGUgICAoS0VSTkVMLjIwNykKICoKICogRGV0ZXJtaW5lIGlmIGEgY2hhcmFjdGVyIGlzIGEgbGVhZCBieXRlLgogKgogKiBQQVJBTVMKICogIHRlc3RjaGFyIFtJXSBDaGFyYWN0ZXIgdG8gdGVzdAogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBpZiB0ZXN0Y2hhciBpcyBhIGxlYWQgYnl0ZSBpbiB0aGUgQW5zaWkgY29kZSBwYWdlLAogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNEQkNTTGVhZEJ5dGUoIEJZVEUgdGVzdGNoYXIgKQp7CiAgICBpZiAoIWFuc2lfY3B0YWJsZSkgcmV0dXJuIEZBTFNFOwogICAgcmV0dXJuIHdpbmVfaXNfZGJjc19sZWFkYnl0ZSggYW5zaV9jcHRhYmxlLCB0ZXN0Y2hhciApOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRDUEluZm8gICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGluZm9ybWF0aW9uIGFib3V0IGEgY29kZSBwYWdlLgogKgogKiBQQVJBTVMKICogIGNvZGVwYWdlIFtJXSBDb2RlIHBhZ2UgbnVtYmVyCiAqICBjcGluZm8gICBbT10gRGVzdGluYXRpb24gZm9yIGNvZGUgcGFnZSBpbmZvcm1hdGlvbgogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBjcGluZm8gaXMgdXBkYXRlZCB3aXRoIHRoZSBpbmZvcm1hdGlvbiBhYm91dCBjb2RlcGFnZS4KICogIEZhaWx1cmU6IEZBTFNFLCBpZiBjb2RlcGFnZSBpcyBpbnZhbGlkIG9yIGNwaW5mbyBpcyBOVUxMLgogKi8KQk9PTCBXSU5BUEkgR2V0Q1BJbmZvKCBVSU5UIGNvZGVwYWdlLCBMUENQSU5GTyBjcGluZm8gKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKCiAgICBpZiAoIWNwaW5mbykKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmICghKHRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBjb2RlcGFnZSApKSkKICAgIHsKICAgICAgICBzd2l0Y2goY29kZXBhZ2UpCiAgICAgICAgewogICAgICAgICAgICBjYXNlIENQX1VURjc6CiAgICAgICAgICAgIGNhc2UgQ1BfVVRGODoKICAgICAgICAgICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMF0gPSAweDNmOwogICAgICAgICAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclsxXSA9IDA7CiAgICAgICAgICAgICAgICBjcGluZm8tPkxlYWRCeXRlWzBdID0gY3BpbmZvLT5MZWFkQnl0ZVsxXSA9IDA7CiAgICAgICAgICAgICAgICBjcGluZm8tPk1heENoYXJTaXplID0gKGNvZGVwYWdlID09IENQX1VURjcpID8gNSA6IDQ7CiAgICAgICAgICAgICAgICByZXR1cm4gVFJVRTsKICAgICAgICB9CgogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CiAgICBpZiAodGFibGUtPmluZm8uZGVmX2NoYXIgJiAweGZmMDApCiAgICB7CiAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclswXSA9ICh0YWJsZS0+aW5mby5kZWZfY2hhciAmIDB4ZmYwMCkgPj4gODsKICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzFdID0gdGFibGUtPmluZm8uZGVmX2NoYXIgJiAweDAwZmY7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclswXSA9IHRhYmxlLT5pbmZvLmRlZl9jaGFyICYgMHhmZjsKICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzFdID0gMDsKICAgIH0KICAgIGlmICgoY3BpbmZvLT5NYXhDaGFyU2l6ZSA9IHRhYmxlLT5pbmZvLmNoYXJfc2l6ZSkgPT0gMikKICAgICAgICBtZW1jcHkoIGNwaW5mby0+TGVhZEJ5dGUsIHRhYmxlLT5kYmNzLmxlYWRfYnl0ZXMsIHNpemVvZihjcGluZm8tPkxlYWRCeXRlKSApOwogICAgZWxzZQogICAgICAgIGNwaW5mby0+TGVhZEJ5dGVbMF0gPSBjcGluZm8tPkxlYWRCeXRlWzFdID0gMDsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRDUEluZm9FeEEgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGV4dGVuZGVkIGluZm9ybWF0aW9uIGFib3V0IGEgY29kZSBwYWdlLgogKgogKiBQQVJBTVMKICogIGNvZGVwYWdlIFtJXSBDb2RlIHBhZ2UgbnVtYmVyCiAqICBkd0ZsYWdzICBbSV0gUmVzZXJ2ZWQsIG11c3QgdG8gMC4KICogIGNwaW5mbyAgIFtPXSBEZXN0aW5hdGlvbiBmb3IgY29kZSBwYWdlIGluZm9ybWF0aW9uCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIGNwaW5mbyBpcyB1cGRhdGVkIHdpdGggdGhlIGluZm9ybWF0aW9uIGFib3V0IGNvZGVwYWdlLgogKiAgRmFpbHVyZTogRkFMU0UsIGlmIGNvZGVwYWdlIGlzIGludmFsaWQgb3IgY3BpbmZvIGlzIE5VTEwuCiAqLwpCT09MIFdJTkFQSSBHZXRDUEluZm9FeEEoIFVJTlQgY29kZXBhZ2UsIERXT1JEIGR3RmxhZ3MsIExQQ1BJTkZPRVhBIGNwaW5mbyApCnsKICAgIENQSU5GT0VYVyBjcGluZm9XOwoKICAgIGlmICghR2V0Q1BJbmZvRXhXKCBjb2RlcGFnZSwgZHdGbGFncywgJmNwaW5mb1cgKSkKICAgICAgcmV0dXJuIEZBTFNFOwoKICAgIC8qIHRoZSBsYXlvdXQgaXMgdGhlIHNhbWUgZXhjZXB0IGZvciBDb2RlUGFnZU5hbWUgKi8KICAgIG1lbWNweShjcGluZm8sICZjcGluZm9XLCBzaXplb2YoQ1BJTkZPRVhBKSk7CiAgICBXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgY3BpbmZvVy5Db2RlUGFnZU5hbWUsIC0xLCBjcGluZm8tPkNvZGVQYWdlTmFtZSwgc2l6ZW9mKGNwaW5mby0+Q29kZVBhZ2VOYW1lKSwgTlVMTCwgTlVMTCk7CiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRDUEluZm9FeFcgICAoS0VSTkVMMzIuQCkKICoKICogVW5pY29kZSB2ZXJzaW9uIG9mIEdldENQSW5mb0V4QS4KICovCkJPT0wgV0lOQVBJIEdldENQSW5mb0V4VyggVUlOVCBjb2RlcGFnZSwgRFdPUkQgZHdGbGFncywgTFBDUElORk9FWFcgY3BpbmZvICkKewogICAgaWYgKCFHZXRDUEluZm8oIGNvZGVwYWdlLCAoTFBDUElORk8pY3BpbmZvICkpCiAgICAgIHJldHVybiBGQUxTRTsKCiAgICBzd2l0Y2goY29kZXBhZ2UpCiAgICB7CiAgICAgICAgY2FzZSBDUF9VVEY3OgogICAgICAgIHsKICAgICAgICAgICAgc3RhdGljIGNvbnN0IFdDSEFSIHV0ZjdbXSA9IHsnVScsJ24nLCdpJywnYycsJ28nLCdkJywnZScsJyAnLCcoJywnVScsJ1QnLCdGJywnLScsJzcnLCcpJywwfTsKCiAgICAgICAgICAgIGNwaW5mby0+Q29kZVBhZ2UgPSBDUF9VVEY3OwogICAgICAgICAgICBjcGluZm8tPlVuaWNvZGVEZWZhdWx0Q2hhciA9IDB4M2Y7CiAgICAgICAgICAgIHN0cmNweVcoY3BpbmZvLT5Db2RlUGFnZU5hbWUsIHV0ZjcpOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CgogICAgICAgIGNhc2UgQ1BfVVRGODoKICAgICAgICB7CiAgICAgICAgICAgIHN0YXRpYyBjb25zdCBXQ0hBUiB1dGY4W10gPSB7J1UnLCduJywnaScsJ2MnLCdvJywnZCcsJ2UnLCcgJywnKCcsJ1UnLCdUJywnRicsJy0nLCc4JywnKScsMH07CgogICAgICAgICAgICBjcGluZm8tPkNvZGVQYWdlID0gQ1BfVVRGODsKICAgICAgICAgICAgY3BpbmZvLT5Vbmljb2RlRGVmYXVsdENoYXIgPSAweDNmOwogICAgICAgICAgICBzdHJjcHlXKGNwaW5mby0+Q29kZVBhZ2VOYW1lLCB1dGY4KTsKICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQoKICAgICAgICBkZWZhdWx0OgogICAgICAgIHsKICAgICAgICAgICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGUgPSBnZXRfY29kZXBhZ2VfdGFibGUoIGNvZGVwYWdlICk7CgogICAgICAgICAgICBjcGluZm8tPkNvZGVQYWdlID0gdGFibGUtPmluZm8uY29kZXBhZ2U7CiAgICAgICAgICAgIGNwaW5mby0+VW5pY29kZURlZmF1bHRDaGFyID0gdGFibGUtPmluZm8uZGVmX3VuaWNvZGVfY2hhcjsKICAgICAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhciggQ1BfQUNQLCAwLCB0YWJsZS0+aW5mby5uYW1lLCAtMSwgY3BpbmZvLT5Db2RlUGFnZU5hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihjcGluZm8tPkNvZGVQYWdlTmFtZSkvc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEVudW1TeXN0ZW1Db2RlUGFnZXNBICAgKEtFUk5FTDMyLkApCiAqCiAqIENhbGwgYSB1c2VyIGRlZmluZWQgZnVuY3Rpb24gZm9yIGV2ZXJ5IGNvZGUgcGFnZSBpbnN0YWxsZWQgb24gdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICAgbHBmbkNvZGVQYWdlRW51bSBbSV0gVXNlciBDT0RFUEFHRV9FTlVNUFJPQyB0byBjYWxsIHdpdGggZWFjaCBmb3VuZCBjb2RlIHBhZ2UKICogICBmbGFncyAgICAgICAgICAgIFtJXSBSZXNlcnZlZCwgc2V0IHRvIDAuCiAqCiAqIFJFVFVSTlMKICogIFRSVUUsIElmIGFsbCBjb2RlIHBhZ2VzIGhhdmUgYmVlbiBlbnVtZXJhdGVkLCBvcgogKiAgRkFMU0UgaWYgbHBmbkNvZGVQYWdlRW51bSByZXR1cm5lZCBGQUxTRSB0byBzdG9wIHRoZSBlbnVtZXJhdGlvbi4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Db2RlUGFnZXNBKCBDT0RFUEFHRV9FTlVNUFJPQ0EgbHBmbkNvZGVQYWdlRW51bSwgRFdPUkQgZmxhZ3MgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIGNoYXIgYnVmZmVyWzEwXTsKICAgIGludCBpbmRleCA9IDA7CgogICAgZm9yICg7OykKICAgIHsKICAgICAgICBpZiAoISh0YWJsZSA9IHdpbmVfY3BfZW51bV90YWJsZSggaW5kZXgrKyApKSkgYnJlYWs7CiAgICAgICAgc3ByaW50ZiggYnVmZmVyLCAiJWQiLCB0YWJsZS0+aW5mby5jb2RlcGFnZSApOwogICAgICAgIGlmICghbHBmbkNvZGVQYWdlRW51bSggYnVmZmVyICkpIGJyZWFrOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEVudW1TeXN0ZW1Db2RlUGFnZXNXICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBFbnVtU3lzdGVtQ29kZVBhZ2VzQS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Db2RlUGFnZXNXKCBDT0RFUEFHRV9FTlVNUFJPQ1cgbHBmbkNvZGVQYWdlRW51bSwgRFdPUkQgZmxhZ3MgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIFdDSEFSIGJ1ZmZlclsxMF0sICpwOwogICAgaW50IHBhZ2UsIGluZGV4ID0gMDsKCiAgICBmb3IgKDs7KQogICAgewogICAgICAgIGlmICghKHRhYmxlID0gd2luZV9jcF9lbnVtX3RhYmxlKCBpbmRleCsrICkpKSBicmVhazsKICAgICAgICBwID0gYnVmZmVyICsgc2l6ZW9mKGJ1ZmZlcikvc2l6ZW9mKFdDSEFSKTsKICAgICAgICAqLS1wID0gMDsKICAgICAgICBwYWdlID0gdGFibGUtPmluZm8uY29kZXBhZ2U7CiAgICAgICAgZG8KICAgICAgICB7CiAgICAgICAgICAgICotLXAgPSAnMCcgKyAocGFnZSAlIDEwKTsKICAgICAgICAgICAgcGFnZSAvPSAxMDsKICAgICAgICB9IHdoaWxlKCBwYWdlICk7CiAgICAgICAgaWYgKCFscGZuQ29kZVBhZ2VFbnVtKCBwICkpIGJyZWFrOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIgICAoS0VSTkVMMzIuQCkKICoKICogQ29udmVydCBhIG11bHRpYnl0ZSBjaGFyYWN0ZXIgc3RyaW5nIGludG8gYSBVbmljb2RlIHN0cmluZy4KICoKICogUEFSQU1TCiAqICAgcGFnZSAgIFtJXSBDb2RlcGFnZSBjaGFyYWN0ZXIgc2V0IHRvIGNvbnZlcnQgZnJvbQogKiAgIGZsYWdzICBbSV0gQ2hhcmFjdGVyIG1hcHBpbmcgZmxhZ3MKICogICBzcmMgICAgW0ldIFNvdXJjZSBzdHJpbmcgYnVmZmVyCiAqICAgc3JjbGVuIFtJXSBMZW5ndGggb2Ygc3JjIChpbiBieXRlcyksIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgIGRzdCAgICBbT10gRGVzdGluYXRpb24gYnVmZmVyCiAqICAgZHN0bGVuIFtJXSBMZW5ndGggb2YgZHN0IChpbiBXQ0hBUnMpLCBvciAwIHRvIGNvbXB1dGUgdGhlIHJlcXVpcmVkIGxlbmd0aAogKgogKiBSRVRVUk5TCiAqICAgU3VjY2VzczogSWYgZHN0bGVuID4gMCwgdGhlIG51bWJlciBvZiBjaGFyYWN0ZXJzIHdyaXR0ZW4gdG8gZHN0LgogKiAgICAgICAgICAgIElmIGRzdGxlbiA9PSAwLCB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgbmVlZGVkIHRvIHBlcmZvcm0gdGhlCiAqICAgICAgICAgICAgY29udmVyc2lvbi4gSW4gYm90aCBjYXNlcyB0aGUgY291bnQgaW5jbHVkZXMgdGhlIHRlcm1pbmF0aW5nIE5VTC4KICogICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4gUG9zc2libGUgZXJyb3JzIGFyZQogKiAgICAgICAgICAgIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIsIGlmIG5vdCBlbm91Z2ggc3BhY2UgaXMgYXZhaWxhYmxlIGluIGRzdAogKiAgICAgICAgICAgIGFuZCBkc3RsZW4gIT0gMDsgRVJST1JfSU5WQUxJRF9QQVJBTUVURVIsICBpZiBhbiBpbnZhbGlkIHBhcmFtZXRlcgogKiAgICAgICAgICAgIGlzIHBhc3NlZCwgYW5kIEVSUk9SX05PX1VOSUNPREVfVFJBTlNMQVRJT04gaWYgbm8gdHJhbnNsYXRpb24gaXMKICogICAgICAgICAgICBwb3NzaWJsZSBmb3Igc3JjLgogKi8KSU5UIFdJTkFQSSBNdWx0aUJ5dGVUb1dpZGVDaGFyKCBVSU5UIHBhZ2UsIERXT1JEIGZsYWdzLCBMUENTVFIgc3JjLCBJTlQgc3JjbGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQV1NUUiBkc3QsIElOVCBkc3RsZW4gKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIGludCByZXQ7CgogICAgaWYgKCFzcmMgfHwgKCFkc3QgJiYgZHN0bGVuKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKHNyY2xlbiA8IDApIHNyY2xlbiA9IHN0cmxlbihzcmMpICsgMTsKCiAgICBzd2l0Y2gocGFnZSkKICAgIHsKICAgIGNhc2UgQ1BfU1lNQk9MOgogICAgICAgIGlmKCBmbGFncykKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfY3BzeW1ib2xfbWJzdG93Y3MoIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBDUF9VVEY3OgogICAgICAgIEZJWE1FKCJVVEYtNyBub3Qgc3VwcG9ydGVkXG4iKTsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0NBTExfTk9UX0lNUExFTUVOVEVEICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICBjYXNlIENQX1VOSVhDUDoKICAgICAgICBpZiAodW5peF9jcHRhYmxlKQogICAgICAgIHsKICAgICAgICAgICAgcmV0ID0gd2luZV9jcF9tYnN0b3djcyggdW5peF9jcHRhYmxlLCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KI2lmZGVmIF9fQVBQTEVfXwogICAgICAgIGZsYWdzIHw9IE1CX0NPTVBPU0lURTsgIC8qIHdvcmsgYXJvdW5kIGJyb2tlbiBNYWMgT1MgWCBmaWxlc3lzdGVtIHRoYXQgZW5mb3JjZXMgZGVjb21wb3NlZCBVbmljb2RlICovCiNlbmRpZgogICAgICAgIC8qIGZhbGwgdGhyb3VnaCAqLwogICAgY2FzZSBDUF9VVEY4OgogICAgICAgIHJldCA9IHdpbmVfdXRmOF9tYnN0b3djcyggZmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgIGJyZWFrOwogICAgZGVmYXVsdDoKICAgICAgICBpZiAoISh0YWJsZSA9IGdldF9jb2RlcGFnZV90YWJsZSggcGFnZSApKSkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfY3BfbWJzdG93Y3MoIHRhYmxlLCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKHJldCA8IDApCiAgICB7CiAgICAgICAgc3dpdGNoKHJldCkKICAgICAgICB7CiAgICAgICAgY2FzZSAtMTogU2V0TGFzdEVycm9yKCBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSICk7IGJyZWFrOwogICAgICAgIGNhc2UgLTI6IFNldExhc3RFcnJvciggRVJST1JfTk9fVU5JQ09ERV9UUkFOU0xBVElPTiApOyBicmVhazsKICAgICAgICB9CiAgICAgICAgcmV0ID0gMDsKICAgIH0KICAgIFRSQUNFKCJjcCAlZCAlcyAtPiAlcywgcmV0ID0gJWRcbiIsCiAgICAgICAgICBwYWdlLCBkZWJ1Z3N0cl9hbihzcmMsIHNyY2xlbiksIGRlYnVnc3RyX3duKGRzdCwgcmV0KSwgcmV0KTsKICAgIHJldHVybiByZXQ7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIFdpZGVDaGFyVG9NdWx0aUJ5dGUgICAoS0VSTkVMMzIuQCkKICoKICogQ29udmVydCBhIFVuaWNvZGUgY2hhcmFjdGVyIHN0cmluZyBpbnRvIGEgbXVsdGlieXRlIHN0cmluZy4KICoKICogUEFSQU1TCiAqICAgcGFnZSAgICBbSV0gQ29kZSBwYWdlIGNoYXJhY3RlciBzZXQgdG8gY29udmVydCB0bwogKiAgIGZsYWdzICAgW0ldIE1hcHBpbmcgRmxhZ3MgKE1CXyBjb25zdGFudHMgZnJvbSAid2lubmxzLmgiKS4KICogICBzcmMgICAgIFtJXSBTb3VyY2Ugc3RyaW5nIGJ1ZmZlcgogKiAgIHNyY2xlbiAgW0ldIExlbmd0aCBvZiBzcmMgKGluIFdDSEFScyksIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgIGRzdCAgICAgW09dIERlc3RpbmF0aW9uIGJ1ZmZlcgogKiAgIGRzdGxlbiAgW0ldIExlbmd0aCBvZiBkc3QgKGluIGJ5dGVzKSwgb3IgMCB0byBjb21wdXRlIHRoZSByZXF1aXJlZCBsZW5ndGgKICogICBkZWZjaGFyIFtJXSBEZWZhdWx0IGNoYXJhY3RlciB0byB1c2UgZm9yIGNvbnZlcnNpb24gaWYgbm8gZXhhY3QKICoJCSAgICBjb252ZXJzaW9uIGNhbiBiZSBtYWRlCiAqICAgdXNlZCAgICBbT10gU2V0IGlmIGRlZmF1bHQgY2hhcmFjdGVyIHdhcyB1c2VkIGluIHRoZSBjb252ZXJzaW9uCiAqCiAqIFJFVFVSTlMKICogICBTdWNjZXNzOiBJZiBkc3RsZW4gPiAwLCB0aGUgbnVtYmVyIG9mIGNoYXJhY3RlcnMgd3JpdHRlbiB0byBkc3QuCiAqICAgICAgICAgICAgSWYgZHN0bGVuID09IDAsIG51bWJlciBvZiBjaGFyYWN0ZXJzIG5lZWRlZCB0byBwZXJmb3JtIHRoZQogKiAgICAgICAgICAgIGNvbnZlcnNpb24uIEluIGJvdGggY2FzZXMgdGhlIGNvdW50IGluY2x1ZGVzIHRoZSB0ZXJtaW5hdGluZyBOVUwuCiAqICAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuIFBvc3NpYmxlIGVycm9ycyBhcmUKICogICAgICAgICAgICBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSLCBpZiBub3QgZW5vdWdoIHNwYWNlIGlzIGF2YWlsYWJsZSBpbiBkc3QKICogICAgICAgICAgICBhbmQgZHN0bGVuICE9IDAsIGFuZCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiwgaWYgYW4gaW52YWxpZAogKiAgICAgICAgICAgIHBhcmFtZXRlciB3YXMgZ2l2ZW4uCiAqLwpJTlQgV0lOQVBJIFdpZGVDaGFyVG9NdWx0aUJ5dGUoIFVJTlQgcGFnZSwgRFdPUkQgZmxhZ3MsIExQQ1dTVFIgc3JjLCBJTlQgc3JjbGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQU1RSIGRzdCwgSU5UIGRzdGxlbiwgTFBDU1RSIGRlZmNoYXIsIEJPT0wgKnVzZWQgKQp7CiAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp0YWJsZTsKICAgIGludCByZXQsIHVzZWRfdG1wOwoKICAgIGlmICghc3JjIHx8ICghZHN0ICYmIGRzdGxlbikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzcmNsZW4gPCAwKSBzcmNsZW4gPSBzdHJsZW5XKHNyYykgKyAxOwoKICAgIHN3aXRjaChwYWdlKQogICAgewogICAgY2FzZSBDUF9TWU1CT0w6CiAgICAgICAgaWYoIGZsYWdzIHx8IGRlZmNoYXIgfHwgdXNlZCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfY3BzeW1ib2xfd2NzdG9tYnMoIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBDUF9VVEY3OgogICAgICAgIEZJWE1FKCJVVEYtNyBub3Qgc3VwcG9ydGVkXG4iKTsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0NBTExfTk9UX0lNUExFTUVOVEVEICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICBjYXNlIENQX1VOSVhDUDoKICAgICAgICBpZiAodW5peF9jcHRhYmxlKQogICAgICAgIHsKICAgICAgICAgICAgcmV0ID0gd2luZV9jcF93Y3N0b21icyggdW5peF9jcHRhYmxlLCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkZWZjaGFyLCB1c2VkID8gJnVzZWRfdG1wIDogTlVMTCApOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgaWYgKHVzZWQpICp1c2VkID0gRkFMU0U7ICAvKiBhbGwgY2hhcnMgYXJlIHZhbGlkIGZvciBVVEYtOCAqLwogICAgICAgIHJldCA9IHdpbmVfdXRmOF93Y3N0b21icyggZmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgIGJyZWFrOwogICAgZGVmYXVsdDoKICAgICAgICBpZiAoISh0YWJsZSA9IGdldF9jb2RlcGFnZV90YWJsZSggcGFnZSApKSkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfY3Bfd2NzdG9tYnMoIHRhYmxlLCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlZmNoYXIsIHVzZWQgPyAmdXNlZF90bXAgOiBOVUxMICk7CiAgICAgICAgaWYgKHVzZWQpICp1c2VkID0gdXNlZF90bXA7CiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKHJldCA8IDApCiAgICB7CiAgICAgICAgc3dpdGNoKHJldCkKICAgICAgICB7CiAgICAgICAgY2FzZSAtMTogU2V0TGFzdEVycm9yKCBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSICk7IGJyZWFrOwogICAgICAgIGNhc2UgLTI6IFNldExhc3RFcnJvciggRVJST1JfTk9fVU5JQ09ERV9UUkFOU0xBVElPTiApOyBicmVhazsKICAgICAgICB9CiAgICAgICAgcmV0ID0gMDsKICAgIH0KICAgIFRSQUNFKCJjcCAlZCAlcyAtPiAlcywgcmV0ID0gJWRcbiIsCiAgICAgICAgICBwYWdlLCBkZWJ1Z3N0cl93bihzcmMsIHNyY2xlbiksIGRlYnVnc3RyX2FuKGRzdCwgcmV0KSwgcmV0KTsKICAgIHJldHVybiByZXQ7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFRocmVhZExvY2FsZSAgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBjdXJyZW50IHRocmVhZHMgbG9jYWxlLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogIFRoZSBMQ0lEIGN1cnJlbnRseSBhc3NvY2lhdGVkIHdpdGggdGhlIGNhbGxpbmcgdGhyZWFkLgogKi8KTENJRCBXSU5BUEkgR2V0VGhyZWFkTG9jYWxlKHZvaWQpCnsKICAgIExDSUQgcmV0ID0gTnRDdXJyZW50VGViKCktPkN1cnJlbnRMb2NhbGU7CiAgICBpZiAoIXJldCkgTnRDdXJyZW50VGViKCktPkN1cnJlbnRMb2NhbGUgPSByZXQgPSBHZXRVc2VyRGVmYXVsdExDSUQoKTsKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBTZXRUaHJlYWRMb2NhbGUgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNldCB0aGUgY3VycmVudCB0aHJlYWRzIGxvY2FsZS4KICoKICogUEFSQU1TCiAqICBsY2lkIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUgdG8gc2V0CiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIFRoZSB0aHJlYWRzIGxvY2FsZSBpcyBzZXQgdG8gbGNpZC4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIFNldFRocmVhZExvY2FsZSggTENJRCBsY2lkICkKewogICAgVFJBQ0UoIigweCUwNFgpXG4iLCBsY2lkKTsKCiAgICBsY2lkID0gQ29udmVydERlZmF1bHRMb2NhbGUobGNpZCk7CgogICAgaWYgKGxjaWQgIT0gR2V0VGhyZWFkTG9jYWxlKCkpCiAgICB7CiAgICAgICAgaWYgKCFJc1ZhbGlkTG9jYWxlKGxjaWQsIExDSURfU1VQUE9SVEVEKSkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgICAgIHJldHVybiBGQUxTRTsKICAgICAgICB9CgogICAgICAgIE50Q3VycmVudFRlYigpLT5DdXJyZW50TG9jYWxlID0gbGNpZDsKICAgICAgICBrZXJuZWxfZ2V0X3RocmVhZF9kYXRhKCktPmNvZGVfcGFnZSA9IGdldF9sY2lkX2NvZGVwYWdlKCBsY2lkICk7CiAgICB9CiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIFNldFRocmVhZFVJTGFuZ3VhZ2UgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNldCB0aGUgY3VycmVudCB0aHJlYWRzIFVJIGxhbmd1YWdlLgogKgogKiBQQVJBTVMKICogIGxhbmdpZCBbSV0gTEFOR0lEIG9mIHRoZSBsYW5ndWFnZSB0byBzZXQsIG9yIDAgdG8gdXNlCiAqICAgICAgICAgICAgIHRoZSBhdmFpbGFibGUgbGFuZ3VhZ2Ugd2hpY2ggaXMgYmVzdCBzdXBwb3J0ZWQKICogICAgICAgICAgICAgZm9yIGNvbnNvbGUgYXBwbGljYXRpb25zCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRoZSByZXR1cm4gdmFsdWUgaXMgdGhlIHNhbWUgYXMgdGhlIGlucHV0IHZhbHVlLgogKiAgRmFpbHVyZTogVGhlIHJldHVybiB2YWx1ZSBkaWZmZXJzIGZyb20gdGhlIGlucHV0IHZhbHVlLgogKiAgICAgICAgICAgVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpMQU5HSUQgV0lOQVBJIFNldFRocmVhZFVJTGFuZ3VhZ2UoIExBTkdJRCBsYW5naWQgKQp7CiAgICBUUkFDRSgiKDB4JTA0eCkgc3R1YiAtIHJldHVybmluZyBzdWNjZXNzXG4iLCBsYW5naWQpOwogICAgcmV0dXJuIGxhbmdpZDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJQ29udmVydERlZmF1bHRMb2NhbGUgKEtFUk5FTDMyLkApCiAqCiAqIENvbnZlcnQgYSBkZWZhdWx0IGxvY2FsZSBpZGVudGlmaWVyIGludG8gYSByZWFsIGlkZW50aWZpZXIuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCBbSV0gTENJRCBpZGVudGlmaWVyIG9mIHRoZSBsb2NhbGUgdG8gY29udmVydAogKgogKiBSRVRVUk5TCiAqICBsY2lkIHVuY2hhbmdlZCwgaWYgbm90IGEgZGVmYXVsdCBsb2NhbGUgb3IgaXRzIHN1Ymxhbmd1YWdlIGlzCiAqICAgbm90IFNVQkxBTkdfTkVVVFJBTC4KICogIEdldFN5c3RlbURlZmF1bHRMQ0lEKCksIGlmIGxjaWQgPT0gTE9DQUxFX1NZU1RFTV9ERUZBVUxULgogKiAgR2V0VXNlckRlZmF1bHRMQ0lEKCksIGlmIGxjaWQgPT0gTE9DQUxFX1VTRVJfREVGQVVMVCBvciBMT0NBTEVfTkVVVFJBTC4KICogIE90aGVyd2lzZSwgbGNpZCB3aXRoIHN1Ymxhbmd1YWdlIGNoYW5nZWQgdG8gU1VCTEFOR19ERUZBVUxULgogKi8KTENJRCBXSU5BUEkgQ29udmVydERlZmF1bHRMb2NhbGUoIExDSUQgbGNpZCApCnsKICAgIExBTkdJRCBsYW5naWQ7CgogICAgc3dpdGNoIChsY2lkKQogICAgewogICAgY2FzZSBMT0NBTEVfU1lTVEVNX0RFRkFVTFQ6CiAgICAgICAgbGNpZCA9IEdldFN5c3RlbURlZmF1bHRMQ0lEKCk7CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIExPQ0FMRV9VU0VSX0RFRkFVTFQ6CiAgICBjYXNlIExPQ0FMRV9ORVVUUkFMOgogICAgICAgIGxjaWQgPSBHZXRVc2VyRGVmYXVsdExDSUQoKTsKICAgICAgICBicmVhazsKICAgIGRlZmF1bHQ6CiAgICAgICAgLyogUmVwbGFjZSBTVUJMQU5HX05FVVRSQUwgd2l0aCBTVUJMQU5HX0RFRkFVTFQgKi8KICAgICAgICBsYW5naWQgPSBMQU5HSURGUk9NTENJRChsY2lkKTsKICAgICAgICBpZiAoU1VCTEFOR0lEKGxhbmdpZCkgPT0gU1VCTEFOR19ORVVUUkFMKQogICAgICAgIHsKICAgICAgICAgIGxhbmdpZCA9IE1BS0VMQU5HSUQoUFJJTUFSWUxBTkdJRChsYW5naWQpLCBTVUJMQU5HX0RFRkFVTFQpOwogICAgICAgICAgbGNpZCA9IE1BS0VMQ0lEKGxhbmdpZCwgU09SVElERlJPTUxDSUQobGNpZCkpOwogICAgICAgIH0KICAgIH0KICAgIHJldHVybiBsY2lkOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSXNWYWxpZExvY2FsZSAgIChLRVJORUwzMi5AKQogKgogKiBEZXRlcm1pbmUgaWYgYSBsb2NhbGUgaXMgdmFsaWQuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgW0ldIExDSUQgb2YgdGhlIGxvY2FsZSB0byBjaGVjawogKiAgZmxhZ3MgW0ldIExDSURfU1VQUE9SVEVEID0gVmFsaWQsIExDSURfSU5TVEFMTEVEID0gVmFsaWQgYW5kIGluc3RhbGxlZCBvbiB0aGUgc3lzdGVtCiAqCiAqIFJFVFVSTlMKICogIFRSVUUsICBpZiBsY2lkIGlzIHZhbGlkLAogKiAgRkFMU0UsIG90aGVyd2lzZS4KICoKICogTk9URVMKICogIFdpbmUgZG9lcyBub3QgY3VycmVudGx5IG1ha2UgdGhlIGRpc3RpbmN0aW9uIGJldHdlZW4gc3VwcG9ydGVkIGFuZCBpbnN0YWxsZWQuIEFsbAogKiAgbGFuZ3VhZ2VzIHN1cHBvcnRlZCBhcmUgaW5zdGFsbGVkIGJ5IGRlZmF1bHQuCiAqLwpCT09MIFdJTkFQSSBJc1ZhbGlkTG9jYWxlKCBMQ0lEIGxjaWQsIERXT1JEIGZsYWdzICkKewogICAgLyogY2hlY2sgaWYgbGFuZ3VhZ2UgaXMgcmVnaXN0ZXJlZCBpbiB0aGUga2VybmVsMzIgcmVzb3VyY2VzICovCiAgICByZXR1cm4gRmluZFJlc291cmNlRXhXKCBrZXJuZWwzMl9oYW5kbGUsIChMUFdTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQQ1dTVFIpTE9DQUxFX0lMQU5HVUFHRSwgTEFOR0lERlJPTUxDSUQobGNpZCkpICE9IDA7Cn0KCgpzdGF0aWMgQk9PTCBDQUxMQkFDSyBlbnVtX2xhbmdfcHJvY19hKCBITU9EVUxFIGhNb2R1bGUsIExQQ1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExPTkdfUFRSIGxQYXJhbSApCnsKICAgIExPQ0FMRV9FTlVNUFJPQ0EgbHBmbkxvY2FsZUVudW0gPSAoTE9DQUxFX0VOVU1QUk9DQSlsUGFyYW07CiAgICBjaGFyIGJ1ZlsyMF07CgogICAgc3ByaW50ZihidWYsICIlMDh4IiwgKFVJTlQpTGFuZ0lEKTsKICAgIHJldHVybiBscGZuTG9jYWxlRW51bSggYnVmICk7Cn0KCnN0YXRpYyBCT09MIENBTExCQUNLIGVudW1fbGFuZ19wcm9jX3coIEhNT0RVTEUgaE1vZHVsZSwgTFBDV1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENXU1RSIG5hbWUsIFdPUkQgTGFuZ0lELCBMT05HX1BUUiBsUGFyYW0gKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgZm9ybWF0V1tdID0geyclJywnMCcsJzgnLCd4JywwfTsKICAgIExPQ0FMRV9FTlVNUFJPQ1cgbHBmbkxvY2FsZUVudW0gPSAoTE9DQUxFX0VOVU1QUk9DVylsUGFyYW07CiAgICBXQ0hBUiBidWZbMjBdOwogICAgc3ByaW50ZlcoIGJ1ZiwgZm9ybWF0VywgKFVJTlQpTGFuZ0lEICk7CiAgICByZXR1cm4gbHBmbkxvY2FsZUVudW0oIGJ1ZiApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtTG9jYWxlc0EgIChLRVJORUwzMi5AKQogKgogKiBDYWxsIGEgdXNlcnMgZnVuY3Rpb24gZm9yIGVhY2ggbG9jYWxlIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIGxwZm5Mb2NhbGVFbnVtIFtJXSBDYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIGZvciBlYWNoIGxvY2FsZQogKiAgZHdGbGFncyAgICAgICAgW0ldIExPQ0FMRV9TVVBQT1JURUQ9QWxsIHN1cHBvcnRlZCwgTE9DQUxFX0lOU1RBTExFRD1JbnN0YWxsZWQgb25seQogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KQk9PTCBXSU5BUEkgRW51bVN5c3RlbUxvY2FsZXNBKCBMT0NBTEVfRU5VTVBST0NBIGxwZm5Mb2NhbGVFbnVtLCBEV09SRCBkd0ZsYWdzICkKewogICAgVFJBQ0UoIiglcCwlMDh4KVxuIiwgbHBmbkxvY2FsZUVudW0sIGR3RmxhZ3MpOwogICAgRW51bVJlc291cmNlTGFuZ3VhZ2VzQSgga2VybmVsMzJfaGFuZGxlLCAoTFBTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQQ1NUUilMT0NBTEVfSUxBTkdVQUdFLCBlbnVtX2xhbmdfcHJvY19hLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExPTkdfUFRSKWxwZm5Mb2NhbGVFbnVtKTsKICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxvY2FsZXNXICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEVudW1TeXN0ZW1Mb2NhbGVzQS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Mb2NhbGVzVyggTE9DQUxFX0VOVU1QUk9DVyBscGZuTG9jYWxlRW51bSwgRFdPUkQgZHdGbGFncyApCnsKICAgIFRSQUNFKCIoJXAsJTA4eClcbiIsIGxwZm5Mb2NhbGVFbnVtLCBkd0ZsYWdzKTsKICAgIEVudW1SZXNvdXJjZUxhbmd1YWdlc1coIGtlcm5lbDMyX2hhbmRsZSwgKExQV1NUUilSVF9TVFJJTkcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBDV1NUUilMT0NBTEVfSUxBTkdVQUdFLCBlbnVtX2xhbmdfcHJvY193LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExPTkdfUFRSKWxwZm5Mb2NhbGVFbnVtKTsKICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBWZXJMYW5ndWFnZU5hbWVBICAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBuYW1lIG9mIGEgbGFuZ3VhZ2UuCiAqCiAqIFBBUkFNUwogKiAgd0xhbmcgIFtJXSBMQU5HSUQgb2YgdGhlIGxhbmd1YWdlCiAqICBzekxhbmcgW09dIERlc3RpbmF0aW9uIGZvciB0aGUgbGFuZ3VhZ2UgbmFtZQogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgc2l6ZSBvZiB0aGUgbGFuZ3VhZ2UgbmFtZS4gSWYgc3pMYW5nIGlzIG5vbi1OVUxMLCBpdCBpcyBmaWxsZWQKICogICAgICAgICAgIHdpdGggdGhlIG5hbWUuCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICoKICovCkRXT1JEIFdJTkFQSSBWZXJMYW5ndWFnZU5hbWVBKCBEV09SRCB3TGFuZywgTFBTVFIgc3pMYW5nLCBEV09SRCBuU2l6ZSApCnsKICAgIHJldHVybiBHZXRMb2NhbGVJbmZvQSggTUFLRUxDSUQod0xhbmcsIFNPUlRfREVGQVVMVCksIExPQ0FMRV9TRU5HTEFOR1VBR0UsIHN6TGFuZywgblNpemUgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgVmVyTGFuZ3VhZ2VOYW1lVyAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBWZXJMYW5ndWFnZU5hbWVBLgogKi8KRFdPUkQgV0lOQVBJIFZlckxhbmd1YWdlTmFtZVcoIERXT1JEIHdMYW5nLCBMUFdTVFIgc3pMYW5nLCBEV09SRCBuU2l6ZSApCnsKICAgIHJldHVybiBHZXRMb2NhbGVJbmZvVyggTUFLRUxDSUQod0xhbmcsIFNPUlRfREVGQVVMVCksIExPQ0FMRV9TRU5HTEFOR1VBR0UsIHN6TGFuZywgblNpemUgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFN0cmluZ1R5cGVXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgR2V0U3RyaW5nVHlwZUEuCiAqLwpCT09MIFdJTkFQSSBHZXRTdHJpbmdUeXBlVyggRFdPUkQgdHlwZSwgTFBDV1NUUiBzcmMsIElOVCBjb3VudCwgTFBXT1JEIGNoYXJ0eXBlICkKewogICAgaWYgKGNvdW50ID09IC0xKSBjb3VudCA9IHN0cmxlblcoc3JjKSArIDE7CiAgICBzd2l0Y2godHlwZSkKICAgIHsKICAgIGNhc2UgQ1RfQ1RZUEUxOgogICAgICAgIHdoaWxlIChjb3VudC0tKSAqY2hhcnR5cGUrKyA9IGdldF9jaGFyX3R5cGVXKCAqc3JjKysgKSAmIDB4ZmZmOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBDVF9DVFlQRTI6CiAgICAgICAgd2hpbGUgKGNvdW50LS0pICpjaGFydHlwZSsrID0gZ2V0X2NoYXJfdHlwZVcoICpzcmMrKyApID4+IDEyOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBDVF9DVFlQRTM6CiAgICB7CiAgICAgICAgV0FSTigiQ1RfQ1RZUEUzOiBzZW1pLXN0dWIuXG4iKTsKICAgICAgICB3aGlsZSAoY291bnQtLSkKICAgICAgICB7CiAgICAgICAgICAgIGludCBjID0gKnNyYzsKICAgICAgICAgICAgV09SRCB0eXBlMSwgdHlwZTMgPSAwOyAvKiBDM19OT1RBUFBMSUNBQkxFICovCgogICAgICAgICAgICB0eXBlMSA9IGdldF9jaGFyX3R5cGVXKCAqc3JjKysgKSAmIDB4ZmZmOwogICAgICAgICAgICAvKiB0cnkgdG8gY29uc3RydWN0IHR5cGUzIGZyb20gdHlwZTEgKi8KICAgICAgICAgICAgaWYodHlwZTEgJiBDMV9TUEFDRSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICBpZih0eXBlMSAmIEMxX0FMUEhBKSB0eXBlMyB8PSBDM19BTFBIQTsKICAgICAgICAgICAgaWYgKChjPj0weDMwQTApJiYoYzw9MHgzMEZGKSkgdHlwZTMgfD0gQzNfS0FUQUtBTkE7CiAgICAgICAgICAgIGlmICgoYz49MHgzMDQwKSYmKGM8PTB4MzA5RikpIHR5cGUzIHw9IEMzX0hJUkFHQU5BOwogICAgICAgICAgICBpZiAoKGM+PTB4NEUwMCkmJihjPD0weDlGQUYpKSB0eXBlMyB8PSBDM19JREVPR1JBUEg7CiAgICAgICAgICAgIGlmICgoYz49MHgwNjAwKSYmKGM8PTB4MDZGRikpIHR5cGUzIHw9IEMzX0tBU0hJREE7CiAgICAgICAgICAgIGlmICgoYz49MHgzMDAwKSYmKGM8PTB4MzAzRikpIHR5cGUzIHw9IEMzX1NZTUJPTDsKCiAgICAgICAgICAgIGlmICgoYz49MHhGRjAwKSYmKGM8PTB4RkY2MCkpIHR5cGUzIHw9IEMzX0ZVTExXSURUSDsKICAgICAgICAgICAgaWYgKChjPj0weEZGMDApJiYoYzw9MHhGRjIwKSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICBpZiAoKGM+PTB4RkYzQikmJihjPD0weEZGNDApKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjVCKSYmKGM8PTB4RkY2MCkpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgaWYgKChjPj0weEZGMjEpJiYoYzw9MHhGRjNBKSkgdHlwZTMgfD0gQzNfQUxQSEE7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjQxKSYmKGM8PTB4RkY1QSkpIHR5cGUzIHw9IEMzX0FMUEhBOwogICAgICAgICAgICBpZiAoKGM+PTB4RkZFMCkmJihjPD0weEZGRTYpKSB0eXBlMyB8PSBDM19GVUxMV0lEVEg7CiAgICAgICAgICAgIGlmICgoYz49MHhGRkUwKSYmKGM8PTB4RkZFNikpIHR5cGUzIHw9IEMzX1NZTUJPTDsKCiAgICAgICAgICAgIGlmICgoYz49MHhGRjYxKSYmKGM8PTB4RkZEQykpIHR5cGUzIHw9IEMzX0hBTEZXSURUSDsKICAgICAgICAgICAgaWYgKChjPj0weEZGNjEpJiYoYzw9MHhGRjY0KSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICBpZiAoKGM+PTB4RkY2NSkmJihjPD0weEZGOUYpKSB0eXBlMyB8PSBDM19LQVRBS0FOQTsKICAgICAgICAgICAgaWYgKChjPj0weEZGNjUpJiYoYzw9MHhGRjlGKSkgdHlwZTMgfD0gQzNfQUxQSEE7CiAgICAgICAgICAgIGlmICgoYz49MHhGRkU4KSYmKGM8PTB4RkZFRSkpIHR5cGUzIHw9IEMzX0hBTEZXSURUSDsKICAgICAgICAgICAgaWYgKChjPj0weEZGRTgpJiYoYzw9MHhGRkVFKSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICAqY2hhcnR5cGUrKyA9IHR5cGUzOwogICAgICAgIH0KICAgICAgICBicmVhazsKICAgIH0KICAgIGRlZmF1bHQ6CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0U3RyaW5nVHlwZUV4VyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEdldFN0cmluZ1R5cGVFeEEuCiAqLwpCT09MIFdJTkFQSSBHZXRTdHJpbmdUeXBlRXhXKCBMQ0lEIGxvY2FsZSwgRFdPUkQgdHlwZSwgTFBDV1NUUiBzcmMsIElOVCBjb3VudCwgTFBXT1JEIGNoYXJ0eXBlICkKewogICAgLyogbG9jYWxlIGlzIGlnbm9yZWQgZm9yIFVuaWNvZGUgKi8KICAgIHJldHVybiBHZXRTdHJpbmdUeXBlVyggdHlwZSwgc3JjLCBjb3VudCwgY2hhcnR5cGUgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFN0cmluZ1R5cGVBICAgIChLRVJORUwzMi5AKQogKgogKiBHZXQgY2hhcmFjdGVyaXN0aWNzIG9mIHRoZSBjaGFyYWN0ZXJzIG1ha2luZyB1cCBhIHN0cmluZy4KICoKICogUEFSQU1TCiAqICBsb2NhbGUgICBbSV0gTG9jYWxlIElkIGZvciB0aGUgc3RyaW5nCiAqICB0eXBlICAgICBbSV0gQ1RfQ1RZUEUxID0gY2xhc3NpZmljYXRpb24sIENUX0NUWVBFMiA9IGRpcmVjdGlvbmFsaXR5LCBDVF9DVFlQRTMgPSB0eXBvZ3JhcGhpYyBpbmZvCiAqICBzcmMgICAgICBbSV0gU3RyaW5nIHRvIGFuYWx5c2UKICogIGNvdW50ICAgIFtJXSBMZW5ndGggb2Ygc3JjIGluIGNoYXJzLCBvciAtMSBpZiBzcmMgaXMgTlVMIHRlcm1pbmF0ZWQKICogIGNoYXJ0eXBlIFtPXSBEZXN0aW5hdGlvbiBmb3IgdGhlIGNhbGN1bGF0ZWQgY2hhcmFjdGVyaXN0aWNzCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuIGNoYXJ0eXBlIGlzIGZpbGxlZCB3aXRoIHRoZSByZXF1ZXN0ZWQgY2hhcmFjdGVyaXN0aWNzIG9mIGVhY2ggY2hhcgogKiAgICAgICAgICAgaW4gc3JjLgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KQk9PTCBXSU5BUEkgR2V0U3RyaW5nVHlwZUEoIExDSUQgbG9jYWxlLCBEV09SRCB0eXBlLCBMUENTVFIgc3JjLCBJTlQgY291bnQsIExQV09SRCBjaGFydHlwZSApCnsKICAgIFVJTlQgY3A7CiAgICBJTlQgY291bnRXOwogICAgTFBXU1RSIHNyY1c7CiAgICBCT09MIHJldCA9IEZBTFNFOwoKICAgIGlmKGNvdW50ID09IC0xKSBjb3VudCA9IHN0cmxlbihzcmMpICsgMTsKCiAgICBpZiAoIShjcCA9IGdldF9sY2lkX2NvZGVwYWdlKCBsb2NhbGUgKSkpCiAgICB7CiAgICAgICAgRklYTUUoIkZvciBsb2NhbGUgJTA0eCB1c2luZyBjdXJyZW50IEFOU0kgY29kZSBwYWdlXG4iLCBsb2NhbGUpOwogICAgICAgIGNwID0gR2V0QUNQKCk7CiAgICB9CgogICAgY291bnRXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihjcCwgMCwgc3JjLCBjb3VudCwgTlVMTCwgMCk7CiAgICBpZigoc3JjVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBjb3VudFcgKiBzaXplb2YoV0NIQVIpKSkpCiAgICB7CiAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihjcCwgMCwgc3JjLCBjb3VudCwgc3JjVywgY291bnRXKTsKICAgIC8qCiAgICAgKiBOT1RFOiB0aGUgdGFyZ2V0IGJ1ZmZlciBoYXMgMSB3b3JkIGZvciBlYWNoIENIQVJBQ1RFUiBpbiB0aGUgc291cmNlCiAgICAgKiBzdHJpbmcsIHdpdGggbXVsdGlieXRlIGNoYXJhY3RlcnMgdGhlcmUgbWF5YmUgYmUgbW9yZSBieXRlcyBpbiBjb3VudAogICAgICogdGhhbiBjaGFyYWN0ZXIgc3BhY2UgaW4gdGhlIGJ1ZmZlciEKICAgICAqLwogICAgICAgIHJldCA9IEdldFN0cmluZ1R5cGVXKHR5cGUsIHNyY1csIGNvdW50VywgY2hhcnR5cGUpOwogICAgICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgfQogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0U3RyaW5nVHlwZUV4QSAgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGNoYXJhY3RlcmlzdGljcyBvZiB0aGUgY2hhcmFjdGVycyBtYWtpbmcgdXAgYSBzdHJpbmcuCiAqCiAqIFBBUkFNUwogKiAgbG9jYWxlICAgW0ldIExvY2FsZSBJZCBmb3IgdGhlIHN0cmluZwogKiAgdHlwZSAgICAgW0ldIENUX0NUWVBFMSA9IGNsYXNzaWZpY2F0aW9uLCBDVF9DVFlQRTIgPSBkaXJlY3Rpb25hbGl0eSwgQ1RfQ1RZUEUzID0gdHlwb2dyYXBoaWMgaW5mbwogKiAgc3JjICAgICAgW0ldIFN0cmluZyB0byBhbmFseXNlCiAqICBjb3VudCAgICBbSV0gTGVuZ3RoIG9mIHNyYyBpbiBjaGFycywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBjaGFydHlwZSBbT10gRGVzdGluYXRpb24gZm9yIHRoZSBjYWxjdWxhdGVkIGNoYXJhY3RlcmlzdGljcwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBjaGFydHlwZSBpcyBmaWxsZWQgd2l0aCB0aGUgcmVxdWVzdGVkIGNoYXJhY3RlcmlzdGljcyBvZiBlYWNoIGNoYXIKICogICAgICAgICAgIGluIHNyYy4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEdldFN0cmluZ1R5cGVFeEEoIExDSUQgbG9jYWxlLCBEV09SRCB0eXBlLCBMUENTVFIgc3JjLCBJTlQgY291bnQsIExQV09SRCBjaGFydHlwZSApCnsKICAgIHJldHVybiBHZXRTdHJpbmdUeXBlQShsb2NhbGUsIHR5cGUsIHNyYywgY291bnQsIGNoYXJ0eXBlKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBMQ01hcFN0cmluZ1cgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBMQ01hcFN0cmluZ0EuCiAqLwpJTlQgV0lOQVBJIExDTWFwU3RyaW5nVyhMQ0lEIGxjaWQsIERXT1JEIGZsYWdzLCBMUENXU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICAgTFBXU1RSIGRzdCwgSU5UIGRzdGxlbikKewogICAgTFBXU1RSIGRzdF9wdHI7CgogICAgaWYgKCFzcmMgfHwgIXNyY2xlbiB8fCBkc3RsZW4gPCAwKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgLyogbXV0dWFsbHkgZXhjbHVzaXZlIGZsYWdzICovCiAgICBpZiAoKGZsYWdzICYgKExDTUFQX0xPV0VSQ0FTRSB8IExDTUFQX1VQUEVSQ0FTRSkpID09IChMQ01BUF9MT1dFUkNBU0UgfCBMQ01BUF9VUFBFUkNBU0UpIHx8CiAgICAgICAgKGZsYWdzICYgKExDTUFQX0hJUkFHQU5BIHwgTENNQVBfS0FUQUtBTkEpKSA9PSAoTENNQVBfSElSQUdBTkEgfCBMQ01BUF9LQVRBS0FOQSkgfHwKICAgICAgICAoZmxhZ3MgJiAoTENNQVBfSEFMRldJRFRIIHwgTENNQVBfRlVMTFdJRFRIKSkgPT0gKExDTUFQX0hBTEZXSURUSCB8IExDTUFQX0ZVTExXSURUSCkgfHwKICAgICAgICAoZmxhZ3MgJiAoTENNQVBfVFJBRElUSU9OQUxfQ0hJTkVTRSB8IExDTUFQX1NJTVBMSUZJRURfQ0hJTkVTRSkpID09IChMQ01BUF9UUkFESVRJT05BTF9DSElORVNFIHwgTENNQVBfU0lNUExJRklFRF9DSElORVNFKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKCFkc3RsZW4pIGRzdCA9IE5VTEw7CgogICAgbGNpZCA9IENvbnZlcnREZWZhdWx0TG9jYWxlKGxjaWQpOwoKICAgIGlmIChmbGFncyAmIExDTUFQX1NPUlRLRVkpCiAgICB7CiAgICAgICAgSU5UIHJldDsKICAgICAgICBpZiAoc3JjID09IGRzdCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQoKICAgICAgICBpZiAoc3JjbGVuIDwgMCkgc3JjbGVuID0gc3RybGVuVyhzcmMpOwoKICAgICAgICBUUkFDRSgiKDB4JTA0eCwweCUwOHgsJXMsJWQsJXAsJWQpXG4iLAogICAgICAgICAgICAgIGxjaWQsIGZsYWdzLCBkZWJ1Z3N0cl93bihzcmMsIHNyY2xlbiksIHNyY2xlbiwgZHN0LCBkc3RsZW4pOwoKICAgICAgICByZXQgPSB3aW5lX2dldF9zb3J0a2V5KGZsYWdzLCBzcmMsIHNyY2xlbiwgKGNoYXIgKilkc3QsIGRzdGxlbik7CiAgICAgICAgaWYgKHJldCA9PSAwKQogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUik7CiAgICAgICAgcmV0dXJuIHJldDsKICAgIH0KCiAgICAvKiBTT1JUX1NUUklOR1NPUlQgbXVzdCBiZSB1c2VkIGV4Y2x1c2l2ZWx5IHdpdGggTENNQVBfU09SVEtFWSAqLwogICAgaWYgKGZsYWdzICYgU09SVF9TVFJJTkdTT1JUKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBpZiAoc3JjbGVuIDwgMCkgc3JjbGVuID0gc3RybGVuVyhzcmMpICsgMTsKCiAgICBUUkFDRSgiKDB4JTA0eCwweCUwOHgsJXMsJWQsJXAsJWQpXG4iLAogICAgICAgICAgbGNpZCwgZmxhZ3MsIGRlYnVnc3RyX3duKHNyYywgc3JjbGVuKSwgc3JjbGVuLCBkc3QsIGRzdGxlbik7CgogICAgaWYgKCFkc3QpIC8qIHJldHVybiByZXF1aXJlZCBzdHJpbmcgbGVuZ3RoICovCiAgICB7CiAgICAgICAgSU5UIGxlbjsKCiAgICAgICAgZm9yIChsZW4gPSAwOyBzcmNsZW47IHNyYysrLCBzcmNsZW4tLSkKICAgICAgICB7CiAgICAgICAgICAgIFdDSEFSIHdjaCA9ICpzcmM7CiAgICAgICAgICAgIC8qIHRlc3RzIHNob3cgdGhhdCB3aW4yayBqdXN0IGlnbm9yZXMgTk9STV9JR05PUkVOT05TUEFDRSwKICAgICAgICAgICAgICogYW5kIHNraXBzIHdoaXRlIHNwYWNlIGFuZCBwdW5jdHVhdGlvbiBjaGFyYWN0ZXJzIGZvcgogICAgICAgICAgICAgKiBOT1JNX0lHTk9SRVNZTUJPTFMuCiAgICAgICAgICAgICAqLwogICAgICAgICAgICBpZiAoKGZsYWdzICYgTk9STV9JR05PUkVTWU1CT0xTKSAmJiAoZ2V0X2NoYXJfdHlwZVcod2NoKSAmIChDMV9QVU5DVCB8IEMxX1NQQUNFKSkpCiAgICAgICAgICAgICAgICBjb250aW51ZTsKICAgICAgICAgICAgbGVuKys7CiAgICAgICAgfQogICAgICAgIHJldHVybiBsZW47CiAgICB9CgogICAgaWYgKGZsYWdzICYgTENNQVBfVVBQRVJDQVNFKQogICAgewogICAgICAgIGZvciAoZHN0X3B0ciA9IGRzdDsgc3JjbGVuICYmIGRzdGxlbjsgc3JjKyssIHNyY2xlbi0tKQogICAgICAgIHsKICAgICAgICAgICAgV0NIQVIgd2NoID0gKnNyYzsKICAgICAgICAgICAgaWYgKChmbGFncyAmIE5PUk1fSUdOT1JFU1lNQk9MUykgJiYgKGdldF9jaGFyX3R5cGVXKHdjaCkgJiAoQzFfUFVOQ1QgfCBDMV9TUEFDRSkpKQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgICpkc3RfcHRyKysgPSB0b3VwcGVyVyh3Y2gpOwogICAgICAgICAgICBkc3RsZW4tLTsKICAgICAgICB9CiAgICB9CiAgICBlbHNlIGlmIChmbGFncyAmIExDTUFQX0xPV0VSQ0FTRSkKICAgIHsKICAgICAgICBmb3IgKGRzdF9wdHIgPSBkc3Q7IHNyY2xlbiAmJiBkc3RsZW47IHNyYysrLCBzcmNsZW4tLSkKICAgICAgICB7CiAgICAgICAgICAgIFdDSEFSIHdjaCA9ICpzcmM7CiAgICAgICAgICAgIGlmICgoZmxhZ3MgJiBOT1JNX0lHTk9SRVNZTUJPTFMpICYmIChnZXRfY2hhcl90eXBlVyh3Y2gpICYgKEMxX1BVTkNUIHwgQzFfU1BBQ0UpKSkKICAgICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgICAqZHN0X3B0cisrID0gdG9sb3dlclcod2NoKTsKICAgICAgICAgICAgZHN0bGVuLS07CiAgICAgICAgfQogICAgfQogICAgZWxzZQogICAgewogICAgICAgIGlmIChzcmMgPT0gZHN0KQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgZm9yIChkc3RfcHRyID0gZHN0OyBzcmNsZW4gJiYgZHN0bGVuOyBzcmMrKywgc3JjbGVuLS0pCiAgICAgICAgewogICAgICAgICAgICBXQ0hBUiB3Y2ggPSAqc3JjOwogICAgICAgICAgICBpZiAoKGZsYWdzICYgTk9STV9JR05PUkVTWU1CT0xTKSAmJiAoZ2V0X2NoYXJfdHlwZVcod2NoKSAmIChDMV9QVU5DVCB8IEMxX1NQQUNFKSkpCiAgICAgICAgICAgICAgICBjb250aW51ZTsKICAgICAgICAgICAgKmRzdF9wdHIrKyA9IHdjaDsKICAgICAgICAgICAgZHN0bGVuLS07CiAgICAgICAgfQogICAgfQoKICAgIGlmIChzcmNsZW4pCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIHJldHVybiBkc3RfcHRyIC0gZHN0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgTENNYXBTdHJpbmdBICAgIChLRVJORUwzMi5AKQogKgogKiBNYXAgY2hhcmFjdGVycyBpbiBhIGxvY2FsZSBzZW5zaXRpdmUgc3RyaW5nLgogKgogKiBQQVJBTVMKICogIGxjaWQgICBbSV0gTENJRCBmb3IgdGhlIGNvbnZlcnNpb24uCiAqICBmbGFncyAgW0ldIEZsYWdzIGNvbnRyb2xsaW5nIHRoZSBtYXBwaW5nIChMQ01BUF8gY29uc3RhbnRzIGZyb20gIndpbm5scy5oIikuCiAqICBzcmMgICAgW0ldIFN0cmluZyB0byBtYXAKICogIHNyY2xlbiBbSV0gTGVuZ3RoIG9mIHNyYyBpbiBjaGFycywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBkc3QgICAgW09dIERlc3RpbmF0aW9uIGZvciBtYXBwZWQgc3RyaW5nCiAqICBkc3RsZW4gW0ldIExlbmd0aCBvZiBkc3QgaW4gY2hhcmFjdGVycwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgbGVuZ3RoIG9mIHRoZSBtYXBwZWQgc3RyaW5nIGluIGRzdCwgaW5jbHVkaW5nIHRoZSBOVUwgdGVybWluYXRvci4KICogIEZhaWx1cmU6IDAuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KSU5UIFdJTkFQSSBMQ01hcFN0cmluZ0EoTENJRCBsY2lkLCBEV09SRCBmbGFncywgTFBDU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICAgTFBTVFIgZHN0LCBJTlQgZHN0bGVuKQp7CiAgICBXQ0hBUiAqYnVmVyA9IE50Q3VycmVudFRlYigpLT5TdGF0aWNVbmljb2RlQnVmZmVyOwogICAgTFBXU1RSIHNyY1csIGRzdFc7CiAgICBJTlQgcmV0ID0gMCwgc3JjbGVuVywgZHN0bGVuVzsKICAgIFVJTlQgbG9jYWxlX2NwID0gQ1BfQUNQOwoKICAgIGlmICghc3JjIHx8ICFzcmNsZW4gfHwgZHN0bGVuIDwgMCkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmICghKGZsYWdzICYgTE9DQUxFX1VTRV9DUF9BQ1ApKSBsb2NhbGVfY3AgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwoKICAgIHNyY2xlblcgPSBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3JjLCBzcmNsZW4sIGJ1ZlcsIDI2MCk7CiAgICBpZiAoc3JjbGVuVykKICAgICAgICBzcmNXID0gYnVmVzsKICAgIGVsc2UKICAgIHsKICAgICAgICBzcmNsZW5XID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHNyYywgc3JjbGVuLCBOVUxMLCAwKTsKICAgICAgICBzcmNXID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY2xlblcgKiBzaXplb2YoV0NIQVIpKTsKICAgICAgICBpZiAoIXNyY1cpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkpOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHNyYywgc3JjbGVuLCBzcmNXLCBzcmNsZW5XKTsKICAgIH0KCiAgICBpZiAoZmxhZ3MgJiBMQ01BUF9TT1JUS0VZKQogICAgewogICAgICAgIGlmIChzcmMgPT0gZHN0KQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgICAgICBnb3RvIG1hcF9zdHJpbmdfZXhpdDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9nZXRfc29ydGtleShmbGFncywgc3JjVywgc3JjbGVuVywgZHN0LCBkc3RsZW4pOwogICAgICAgIGlmIChyZXQgPT0gMCkKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIpOwogICAgICAgIGdvdG8gbWFwX3N0cmluZ19leGl0OwogICAgfQoKICAgIGlmIChmbGFncyAmIFNPUlRfU1RSSU5HU09SVCkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CiAgICB9CgogICAgZHN0bGVuVyA9IExDTWFwU3RyaW5nVyhsY2lkLCBmbGFncywgc3JjVywgc3JjbGVuVywgTlVMTCwgMCk7CiAgICBpZiAoIWRzdGxlblcpCiAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CgogICAgZHN0VyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBkc3RsZW5XICogc2l6ZW9mKFdDSEFSKSk7CiAgICBpZiAoIWRzdFcpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICBnb3RvIG1hcF9zdHJpbmdfZXhpdDsKICAgIH0KCiAgICBMQ01hcFN0cmluZ1cobGNpZCwgZmxhZ3MsIHNyY1csIHNyY2xlblcsIGRzdFcsIGRzdGxlblcpOwogICAgcmV0ID0gV2lkZUNoYXJUb011bHRpQnl0ZShsb2NhbGVfY3AsIDAsIGRzdFcsIGRzdGxlblcsIGRzdCwgZHN0bGVuLCBOVUxMLCBOVUxMKTsKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGRzdFcpOwoKbWFwX3N0cmluZ19leGl0OgogICAgaWYgKHNyY1cgIT0gYnVmVykgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3JjVyk7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRm9sZFN0cmluZ0EgICAgKEtFUk5FTDMyLkApCiAqCiAqIE1hcCBjaGFyYWN0ZXJzIGluIGEgc3RyaW5nLgogKgogKiBQQVJBTVMKICogIGR3RmxhZ3MgW0ldIEZsYWdzIGNvbnRyb2xsaW5nIGNoYXJzIHRvIG1hcCAoTUFQXyBjb25zdGFudHMgZnJvbSAid2lubmxzLmgiKQogKiAgc3JjICAgICBbSV0gU3RyaW5nIHRvIG1hcAogKiAgc3JjbGVuICBbSV0gTGVuZ3RoIG9mIHNyYywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBkc3QgICAgIFtPXSBEZXN0aW5hdGlvbiBmb3IgbWFwcGVkIHN0cmluZwogKiAgZHN0bGVuICBbSV0gTGVuZ3RoIG9mIGRzdCwgb3IgMCB0byBmaW5kIHRoZSByZXF1aXJlZCBsZW5ndGggZm9yIHRoZSBtYXBwZWQgc3RyaW5nCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRoZSBsZW5ndGggb2YgdGhlIHN0cmluZyB3cml0dGVuIHRvIGRzdCwgaW5jbHVkaW5nIHRoZSB0ZXJtaW5hdGluZyBOVUwuIElmCiAqICAgICAgICAgICBkc3RsZW4gaXMgMCwgdGhlIHZhbHVlIHJldHVybmVkIGlzIHRoZSBzYW1lLCBidXQgbm90aGluZyBpcyB3cml0dGVuIHRvIGRzdCwKICogICAgICAgICAgIGFuZCBkc3QgbWF5IGJlIE5VTEwuCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCklOVCBXSU5BUEkgRm9sZFN0cmluZ0EoRFdPUkQgZHdGbGFncywgTFBDU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICBMUFNUUiBkc3QsIElOVCBkc3RsZW4pCnsKICAgIElOVCByZXQgPSAwLCBzcmNsZW5XID0gMDsKICAgIFdDSEFSICpzcmNXID0gTlVMTCwgKmRzdFcgPSBOVUxMOwoKICAgIGlmICghc3JjIHx8ICFzcmNsZW4gfHwgZHN0bGVuIDwgMCB8fCAoZHN0bGVuICYmICFkc3QpIHx8IHNyYyA9PSBkc3QpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBzcmNsZW5XID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIGR3RmxhZ3MgJiBNQVBfQ09NUE9TSVRFID8gTUJfQ09NUE9TSVRFIDogMCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNyYywgc3JjbGVuLCBOVUxMLCAwKTsKICAgIHNyY1cgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3JjbGVuVyAqIHNpemVvZihXQ0hBUikpOwoKICAgIGlmICghc3JjVykKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkpOwogICAgICAgIGdvdG8gRm9sZFN0cmluZ0FfZXhpdDsKICAgIH0KCiAgICBNdWx0aUJ5dGVUb1dpZGVDaGFyKENQX0FDUCwgZHdGbGFncyAmIE1BUF9DT01QT1NJVEUgPyBNQl9DT01QT1NJVEUgOiAwLAogICAgICAgICAgICAgICAgICAgICAgICBzcmMsIHNyY2xlbiwgc3JjVywgc3JjbGVuVyk7CgogICAgZHdGbGFncyA9IChkd0ZsYWdzICYgfk1BUF9QUkVDT01QT1NFRCkgfCBNQVBfRk9MRENaT05FOwoKICAgIHJldCA9IEZvbGRTdHJpbmdXKGR3RmxhZ3MsIHNyY1csIHNyY2xlblcsIE5VTEwsIDApOwogICAgaWYgKHJldCAmJiBkc3RsZW4pCiAgICB7CiAgICAgICAgZHN0VyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCByZXQgKiBzaXplb2YoV0NIQVIpKTsKCiAgICAgICAgaWYgKCFkc3RXKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICAgICAgZ290byBGb2xkU3RyaW5nQV9leGl0OwogICAgICAgIH0KCiAgICAgICAgcmV0ID0gRm9sZFN0cmluZ1coZHdGbGFncywgc3JjVywgc3JjbGVuVywgZHN0VywgcmV0KTsKICAgICAgICBpZiAoIVdpZGVDaGFyVG9NdWx0aUJ5dGUoQ1BfQUNQLCAwLCBkc3RXLCByZXQsIGRzdCwgZHN0bGVuLCBOVUxMLCBOVUxMKSkKICAgICAgICB7CiAgICAgICAgICAgIHJldCA9IDA7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSKTsKICAgICAgICB9CiAgICB9CgogICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgZHN0Vyk7CgpGb2xkU3RyaW5nQV9leGl0OgogICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3JjVyk7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRm9sZFN0cmluZ1cgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBGb2xkU3RyaW5nQS4KICovCklOVCBXSU5BUEkgRm9sZFN0cmluZ1coRFdPUkQgZHdGbGFncywgTFBDV1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgTFBXU1RSIGRzdCwgSU5UIGRzdGxlbikKewogICAgaW50IHJldDsKCiAgICBzd2l0Y2ggKGR3RmxhZ3MgJiAoTUFQX0NPTVBPU0lURXxNQVBfUFJFQ09NUE9TRUR8TUFQX0VYUEFORF9MSUdBVFVSRVMpKQogICAgewogICAgY2FzZSAwOgogICAgICAgIGlmIChkd0ZsYWdzKQogICAgICAgICAgYnJlYWs7CiAgICAgICAgLyogRmFsbCB0aHJvdWdoIGZvciBkd0ZsYWdzID09IDAgKi8KICAgIGNhc2UgTUFQX1BSRUNPTVBPU0VEfE1BUF9DT01QT1NJVEU6CiAgICBjYXNlIE1BUF9QUkVDT01QT1NFRHxNQVBfRVhQQU5EX0xJR0FUVVJFUzoKICAgIGNhc2UgTUFQX0NPTVBPU0lURXxNQVBfRVhQQU5EX0xJR0FUVVJFUzoKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKCFzcmMgfHwgIXNyY2xlbiB8fCBkc3RsZW4gPCAwIHx8IChkc3RsZW4gJiYgIWRzdCkgfHwgc3JjID09IGRzdCkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIHJldCA9IHdpbmVfZm9sZF9zdHJpbmcoZHdGbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuKTsKICAgIGlmICghcmV0KQogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSKTsKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIENvbXBhcmVTdHJpbmdXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgQ29tcGFyZVN0cmluZ0EuCiAqLwpJTlQgV0lOQVBJIENvbXBhcmVTdHJpbmdXKExDSUQgbGNpZCwgRFdPUkQgc3R5bGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDV1NUUiBzdHIxLCBJTlQgbGVuMSwgTFBDV1NUUiBzdHIyLCBJTlQgbGVuMikKewogICAgSU5UIHJldDsKCiAgICBpZiAoIXN0cjEgfHwgIXN0cjIpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBpZiggc3R5bGUgJiB+KE5PUk1fSUdOT1JFQ0FTRXxOT1JNX0lHTk9SRU5PTlNQQUNFfE5PUk1fSUdOT1JFU1lNQk9MU3wKICAgICAgICBTT1JUX1NUUklOR1NPUlR8Tk9STV9JR05PUkVLQU5BVFlQRXxOT1JNX0lHTk9SRVdJRFRIfExPQ0FMRV9VU0VfQ1BfQUNQfDB4MTAwMDAwMDApICkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgLyogdGhpcyBzdHlsZSBpcyByZWxhdGVkIHRvIGRpYWNyaXRpY3MgaW4gQXJhYmljLCBKYXBhbmVzZSwgYW5kIEhlYnJldyAqLwogICAgaWYgKHN0eWxlICYgMHgxMDAwMDAwMCkKICAgICAgICBXQVJOKCJJZ25vcmluZyB1bmtub3duIHN0eWxlIDB4MTAwMDAwMDBcbiIpOwoKICAgIGlmIChsZW4xIDwgMCkgbGVuMSA9IHN0cmxlblcoc3RyMSk7CiAgICBpZiAobGVuMiA8IDApIGxlbjIgPSBzdHJsZW5XKHN0cjIpOwoKICAgIHJldCA9IHdpbmVfY29tcGFyZV9zdHJpbmcoc3R5bGUsIHN0cjEsIGxlbjEsIHN0cjIsIGxlbjIpOwoKICAgIGlmIChyZXQpIC8qIG5lZWQgdG8gdHJhbnNsYXRlIHJlc3VsdCAqLwogICAgICAgIHJldHVybiAocmV0IDwgMCkgPyBDU1RSX0xFU1NfVEhBTiA6IENTVFJfR1JFQVRFUl9USEFOOwogICAgcmV0dXJuIENTVFJfRVFVQUw7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIENvbXBhcmVTdHJpbmdBICAgIChLRVJORUwzMi5AKQogKgogKiBDb21wYXJlIHR3byBsb2NhbGUgc2Vuc2l0aXZlIHN0cmluZ3MuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCAgW0ldIExDSUQgZm9yIHRoZSBjb21wYXJpc29uCiAqICBzdHlsZSBbSV0gRmxhZ3MgZm9yIHRoZSBjb21wYXJpc29uIChOT1JNXyBjb25zdGFudHMgZnJvbSAid2lubmxzLmgiKS4KICogIHN0cjEgIFtJXSBGaXJzdCBzdHJpbmcgdG8gY29tcGFyZQogKiAgbGVuMSAgW0ldIExlbmd0aCBvZiBzdHIxLCBvciAtMSBpZiBzdHIxIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBzdHIyICBbSV0gU2Vjb25kIHN0cmluZyB0byBjb21wYXJlCiAqICBsZW4yICBbSV0gTGVuZ3RoIG9mIHN0cjIsIG9yIC0xIGlmIHN0cjIgaXMgTlVMIHRlcm1pbmF0ZWQKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogQ1NUUl9MRVNTX1RIQU4sIENTVFJfRVFVQUwgb3IgQ1NUUl9HUkVBVEVSX1RIQU4gZGVwZW5kaW5nIG9uIHdoZXRoZXIKICogICAgICAgICAgIHN0cjIgaXMgbGVzcyB0aGFuLCBlcXVhbCB0byBvciBncmVhdGVyIHRoYW4gc3RyMSByZXNwZWN0aXZlbHkuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpJTlQgV0lOQVBJIENvbXBhcmVTdHJpbmdBKExDSUQgbGNpZCwgRFdPUkQgc3R5bGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDU1RSIHN0cjEsIElOVCBsZW4xLCBMUENTVFIgc3RyMiwgSU5UIGxlbjIpCnsKICAgIFdDSEFSICpidWYxVyA9IE50Q3VycmVudFRlYigpLT5TdGF0aWNVbmljb2RlQnVmZmVyOwogICAgV0NIQVIgKmJ1ZjJXID0gYnVmMVcgKyAxMzA7CiAgICBMUFdTVFIgc3RyMVcsIHN0cjJXOwogICAgSU5UIGxlbjFXLCBsZW4yVywgcmV0OwogICAgVUlOVCBsb2NhbGVfY3AgPSBDUF9BQ1A7CgogICAgaWYgKCFzdHIxIHx8ICFzdHIyKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CiAgICBpZiAobGVuMSA8IDApIGxlbjEgPSBzdHJsZW4oc3RyMSk7CiAgICBpZiAobGVuMiA8IDApIGxlbjIgPSBzdHJsZW4oc3RyMik7CgogICAgaWYgKCEoc3R5bGUgJiBMT0NBTEVfVVNFX0NQX0FDUCkpIGxvY2FsZV9jcCA9IGdldF9sY2lkX2NvZGVwYWdlKCBsY2lkICk7CgogICAgbGVuMVcgPSBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMSwgbGVuMSwgYnVmMVcsIDEzMCk7CiAgICBpZiAobGVuMVcpCiAgICAgICAgc3RyMVcgPSBidWYxVzsKICAgIGVsc2UKICAgIHsKICAgICAgICBsZW4xVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIxLCBsZW4xLCBOVUxMLCAwKTsKICAgICAgICBzdHIxVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBsZW4xVyAqIHNpemVvZihXQ0hBUikpOwogICAgICAgIGlmICghc3RyMVcpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkpOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjEsIGxlbjEsIHN0cjFXLCBsZW4xVyk7CiAgICB9CiAgICBsZW4yVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIyLCBsZW4yLCBidWYyVywgMTMwKTsKICAgIGlmIChsZW4yVykKICAgICAgICBzdHIyVyA9IGJ1ZjJXOwogICAgZWxzZQogICAgewogICAgICAgIGxlbjJXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjIsIGxlbjIsIE5VTEwsIDApOwogICAgICAgIHN0cjJXID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIGxlbjJXICogc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgaWYgKCFzdHIyVykKICAgICAgICB7CiAgICAgICAgICAgIGlmIChzdHIxVyAhPSBidWYxVykgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3RyMVcpOwogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfTk9UX0VOT1VHSF9NRU1PUlkpOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjIsIGxlbjIsIHN0cjJXLCBsZW4yVyk7CiAgICB9CgogICAgcmV0ID0gQ29tcGFyZVN0cmluZ1cobGNpZCwgc3R5bGUsIHN0cjFXLCBsZW4xVywgc3RyMlcsIGxlbjJXKTsKCiAgICBpZiAoc3RyMVcgIT0gYnVmMVcpIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHN0cjFXKTsKICAgIGlmIChzdHIyVyAhPSBidWYyVykgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc3RyMlcpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGxzdHJjbXAgICAgIChLRVJORUwzMi5AKQogKiAgICAgICAgICAgbHN0cmNtcEEgICAgKEtFUk5FTDMyLkApCiAqCiAqIENvbXBhcmUgdHdvIHN0cmluZ3MgdXNpbmcgdGhlIGN1cnJlbnQgdGhyZWFkIGxvY2FsZS4KICoKICogUEFSQU1TCiAqICBzdHIxICBbSV0gRmlyc3Qgc3RyaW5nIHRvIGNvbXBhcmUKICogIHN0cjIgIFtJXSBTZWNvbmQgc3RyaW5nIHRvIGNvbXBhcmUKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogQSBudW1iZXIgbGVzcyB0aGFuLCBlcXVhbCB0byBvciBncmVhdGVyIHRoYW4gMCBkZXBlbmRpbmcgb24gd2hldGhlcgogKiAgICAgICAgICAgc3RyMiBpcyBsZXNzIHRoYW4sIGVxdWFsIHRvIG9yIGdyZWF0ZXIgdGhhbiBzdHIxIHJlc3BlY3RpdmVseS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCmludCBXSU5BUEkgbHN0cmNtcEEoTFBDU1RSIHN0cjEsIExQQ1NUUiBzdHIyKQp7CiAgICBpbnQgcmV0OwogICAgCiAgICBpZiAoKHN0cjEgPT0gTlVMTCkgJiYgKHN0cjIgPT0gTlVMTCkpIHJldHVybiAwOwogICAgaWYgKHN0cjEgPT0gTlVMTCkgcmV0dXJuIC0xOwogICAgaWYgKHN0cjIgPT0gTlVMTCkgcmV0dXJuIDE7CgogICAgcmV0ID0gQ29tcGFyZVN0cmluZ0EoR2V0VGhyZWFkTG9jYWxlKCksIExPQ0FMRV9VU0VfQ1BfQUNQLCBzdHIxLCAtMSwgc3RyMiwgLTEpOwogICAgaWYgKHJldCkgcmV0IC09IDI7CiAgICAKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBsc3RyY21waSAgICAgKEtFUk5FTDMyLkApCiAqICAgICAgICAgICBsc3RyY21waUEgICAgKEtFUk5FTDMyLkApCiAqCiAqIENvbXBhcmUgdHdvIHN0cmluZ3MgdXNpbmcgdGhlIGN1cnJlbnQgdGhyZWFkIGxvY2FsZSwgaWdub3JpbmcgY2FzZS4KICoKICogUEFSQU1TCiAqICBzdHIxICBbSV0gRmlyc3Qgc3RyaW5nIHRvIGNvbXBhcmUKICogIHN0cjIgIFtJXSBTZWNvbmQgc3RyaW5nIHRvIGNvbXBhcmUKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogQSBudW1iZXIgbGVzcyB0aGFuLCBlcXVhbCB0byBvciBncmVhdGVyIHRoYW4gMCBkZXBlbmRpbmcgb24gd2hldGhlcgogKiAgICAgICAgICAgc3RyMiBpcyBsZXNzIHRoYW4sIGVxdWFsIHRvIG9yIGdyZWF0ZXIgdGhhbiBzdHIxIHJlc3BlY3RpdmVseS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCmludCBXSU5BUEkgbHN0cmNtcGlBKExQQ1NUUiBzdHIxLCBMUENTVFIgc3RyMikKewogICAgaW50IHJldDsKICAgIAogICAgaWYgKChzdHIxID09IE5VTEwpICYmIChzdHIyID09IE5VTEwpKSByZXR1cm4gMDsKICAgIGlmIChzdHIxID09IE5VTEwpIHJldHVybiAtMTsKICAgIGlmIChzdHIyID09IE5VTEwpIHJldHVybiAxOwoKICAgIHJldCA9IENvbXBhcmVTdHJpbmdBKEdldFRocmVhZExvY2FsZSgpLCBOT1JNX0lHTk9SRUNBU0V8TE9DQUxFX1VTRV9DUF9BQ1AsIHN0cjEsIC0xLCBzdHIyLCAtMSk7CiAgICBpZiAocmV0KSByZXQgLT0gMjsKICAgIAogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGxzdHJjbXBXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgbHN0cmNtcEEuCiAqLwppbnQgV0lOQVBJIGxzdHJjbXBXKExQQ1dTVFIgc3RyMSwgTFBDV1NUUiBzdHIyKQp7CiAgICBpbnQgcmV0OwoKICAgIGlmICgoc3RyMSA9PSBOVUxMKSAmJiAoc3RyMiA9PSBOVUxMKSkgcmV0dXJuIDA7CiAgICBpZiAoc3RyMSA9PSBOVUxMKSByZXR1cm4gLTE7CiAgICBpZiAoc3RyMiA9PSBOVUxMKSByZXR1cm4gMTsKCiAgICByZXQgPSBDb21wYXJlU3RyaW5nVyhHZXRUaHJlYWRMb2NhbGUoKSwgMCwgc3RyMSwgLTEsIHN0cjIsIC0xKTsKICAgIGlmIChyZXQpIHJldCAtPSAyOwogICAgCiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgbHN0cmNtcGlXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgbHN0cmNtcGlBLgogKi8KaW50IFdJTkFQSSBsc3RyY21waVcoTFBDV1NUUiBzdHIxLCBMUENXU1RSIHN0cjIpCnsKICAgIGludCByZXQ7CiAgICAKICAgIGlmICgoc3RyMSA9PSBOVUxMKSAmJiAoc3RyMiA9PSBOVUxMKSkgcmV0dXJuIDA7CiAgICBpZiAoc3RyMSA9PSBOVUxMKSByZXR1cm4gLTE7CiAgICBpZiAoc3RyMiA9PSBOVUxMKSByZXR1cm4gMTsKCiAgICByZXQgPSBDb21wYXJlU3RyaW5nVyhHZXRUaHJlYWRMb2NhbGUoKSwgTk9STV9JR05PUkVDQVNFLCBzdHIxLCAtMSwgc3RyMiwgLTEpOwogICAgaWYgKHJldCkgcmV0IC09IDI7CiAgICAKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUxPQ0FMRV9Jbml0CiAqLwp2b2lkIExPQ0FMRV9Jbml0KHZvaWQpCnsKICAgIGV4dGVybiB2b2lkIF9fd2luZV9pbml0X2NvZGVwYWdlcyggY29uc3QgdW5pb24gY3B0YWJsZSAqYW5zaV9jcCwgY29uc3QgdW5pb24gY3B0YWJsZSAqb2VtX2NwLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB1bmlvbiBjcHRhYmxlICp1bml4X2NwICk7CgogICAgVUlOVCBhbnNpX2NwID0gMTI1Miwgb2VtX2NwID0gNDM3LCBtYWNfY3AgPSAxMDAwMCwgdW5peF9jcDsKCiNpZmRlZiBfX0FQUExFX18KICAgIC8qIE1hY09TIGRvZXNuJ3Qgc2V0IHRoZSBsb2NhbGUgZW52aXJvbm1lbnQgdmFyaWFibGVzIHNvIHdlIGhhdmUgdG8gZG8gaXQgb3Vyc2VsdmVzICovCiAgICBDRkFycmF5UmVmIHByZWZlcnJlZF9sb2NhbGVzLCBhbGxfbG9jYWxlczsKICAgIENGU3RyaW5nUmVmIHVzZXJfbGFuZ3VhZ2Vfc3RyaW5nX3JlZiA9IE5VTEw7CiAgICBjaGFyIHVzZXJfbG9jYWxlWzUwXTsKCiAgICBDRkxvY2FsZVJlZiB1c2VyX2xvY2FsZV9yZWYgPSBDRkxvY2FsZUNvcHlDdXJyZW50KCk7CiAgICBDRlN0cmluZ1JlZiB1c2VyX2xvY2FsZV9zdHJpbmdfcmVmID0gQ0ZMb2NhbGVHZXRJZGVudGlmaWVyKCB1c2VyX2xvY2FsZV9yZWYgKTsKCiAgICBDRlN0cmluZ0dldENTdHJpbmcoIHVzZXJfbG9jYWxlX3N0cmluZ19yZWYsIHVzZXJfbG9jYWxlLCBzaXplb2YodXNlcl9sb2NhbGUpLCBrQ0ZTdHJpbmdFbmNvZGluZ1VURjggKTsKICAgIENGUmVsZWFzZSggdXNlcl9sb2NhbGVfcmVmICk7CiAgICBpZiAoIXN0cmNociggdXNlcl9sb2NhbGUsICcuJyApKSBzdHJjYXQoIHVzZXJfbG9jYWxlLCAiLlVURi04IiApOwogICAgdW5peF9jcCA9IENQX1VURjg7ICAvKiBkZWZhdWx0IHRvIHV0Zi04IGV2ZW4gaWYgd2UgZG9uJ3QgZ2V0IGEgdmFsaWQgbG9jYWxlICovCiAgICBzZXRlbnYoICJMQU5HIiwgdXNlcl9sb2NhbGUsIDAgKTsKICAgIFRSQUNFKCAic2V0dGluZyBsb2NhbGUgdG8gJyVzJ1xuIiwgdXNlcl9sb2NhbGUgKTsKCiAgICAvKiBXZSBzdGlsbCB3YW50IHRvIHNldCB0aGUgcmV0cmlldmUgdGhlIHByZWZlcnJlZCBsYW5ndWFnZSBhcyBjaG9zZW4gaW4KICAgICAgIFN5c3RlbSBQcmVmZXJlbmNlcy5hcHAsIGJlY2F1c2UgaXQgY2FuIGRpZmZlciBmcm9tIENGTG9jYWxlQ29weUN1cnJlbnQoKS4KICAgICovCiAgICBhbGxfbG9jYWxlcyA9IENGTG9jYWxlQ29weUF2YWlsYWJsZUxvY2FsZUlkZW50aWZpZXJzKCk7CiAgICBwcmVmZXJyZWRfbG9jYWxlcyA9IENGQnVuZGxlQ29weUxvY2FsaXphdGlvbnNGb3JQcmVmZXJlbmNlcyggYWxsX2xvY2FsZXMsIE5VTEwgKTsKICAgIGlmIChwcmVmZXJyZWRfbG9jYWxlcyAmJiBDRkFycmF5R2V0Q291bnQoIHByZWZlcnJlZF9sb2NhbGVzICkpCiAgICAgICAgdXNlcl9sYW5ndWFnZV9zdHJpbmdfcmVmID0gQ0ZBcnJheUdldFZhbHVlQXRJbmRleCggcHJlZmVycmVkX2xvY2FsZXMsIDAgKTsKICAgIENGUmVsZWFzZSggYWxsX2xvY2FsZXMgKTsKI2VuZGlmIC8qIF9fQVBQTEVfXyAqLwoKICAgIHNldGxvY2FsZSggTENfQUxMLCAiIiApOwoKICAgIHVuaXhfY3AgPSBzZXR1cF91bml4X2xvY2FsZXMoKTsKICAgIGlmICghbGNpZF9MQ19NRVNTQUdFUykgbGNpZF9MQ19NRVNTQUdFUyA9IGxjaWRfTENfQ1RZUEU7CgojaWZkZWYgX19BUFBMRV9fCiAgICAvKiBPdmVycmlkZSBsY2lkX0xDX01FU1NBR0VTIHdpdGggdXNlcl9sYW5ndWFnZSBpZiBMQ19NRVNTQUdFUyBpcyBzZXQgdG8gZGVmYXVsdCAqLwogICAgaWYgKGxjaWRfTENfTUVTU0FHRVMgPT0gbGNpZF9MQ19DVFlQRSAmJiB1c2VyX2xhbmd1YWdlX3N0cmluZ19yZWYpCiAgICB7CiAgICAgICAgc3RydWN0IGxvY2FsZV9uYW1lIGxvY2FsZV9uYW1lOwogICAgICAgIFdDSEFSIGJ1ZmZlclsxMjhdOwogICAgICAgIENGU3RyaW5nR2V0Q1N0cmluZyggdXNlcl9sYW5ndWFnZV9zdHJpbmdfcmVmLCB1c2VyX2xvY2FsZSwgc2l6ZW9mKHVzZXJfbG9jYWxlKSwga0NGU3RyaW5nRW5jb2RpbmdVVEY4ICk7CiAgICAgICAgc3RyY3B5bkF0b1coIGJ1ZmZlciwgdXNlcl9sb2NhbGUsIHNpemVvZihidWZmZXIpL3NpemVvZihXQ0hBUikgKTsKICAgICAgICBwYXJzZV9sb2NhbGVfbmFtZSggYnVmZmVyLCAmbG9jYWxlX25hbWUgKTsKICAgICAgICBsY2lkX0xDX01FU1NBR0VTID0gbG9jYWxlX25hbWUubGNpZDsKICAgICAgICBUUkFDRSggInNldHRpbmcgbGNpZF9MQ19NRVNTQUdFUyB0byAnJXMnXG4iLCB1c2VyX2xvY2FsZSApOwogICAgfQogICAgaWYgKHByZWZlcnJlZF9sb2NhbGVzKQogICAgICAgIENGUmVsZWFzZSggcHJlZmVycmVkX2xvY2FsZXMgKTsKI2VuZGlmCgogICAgTnRTZXREZWZhdWx0VUlMYW5ndWFnZSggTEFOR0lERlJPTUxDSUQobGNpZF9MQ19NRVNTQUdFUykgKTsKICAgIE50U2V0RGVmYXVsdExvY2FsZSggVFJVRSwgbGNpZF9MQ19NRVNTQUdFUyApOwogICAgTnRTZXREZWZhdWx0TG9jYWxlKCBGQUxTRSwgbGNpZF9MQ19DVFlQRSApOwoKICAgIGFuc2lfY3AgPSBnZXRfbGNpZF9jb2RlcGFnZSggTE9DQUxFX1VTRVJfREVGQVVMVCApOwogICAgR2V0TG9jYWxlSW5mb1coIExPQ0FMRV9VU0VSX0RFRkFVTFQsIExPQ0FMRV9JREVGQVVMVE1BQ0NPREVQQUdFIHwgTE9DQUxFX1JFVFVSTl9OVU1CRVIsCiAgICAgICAgICAgICAgICAgICAgKExQV1NUUikmbWFjX2NwLCBzaXplb2YobWFjX2NwKS9zaXplb2YoV0NIQVIpICk7CiAgICBHZXRMb2NhbGVJbmZvVyggTE9DQUxFX1VTRVJfREVGQVVMVCwgTE9DQUxFX0lERUZBVUxUQ09ERVBBR0UgfCBMT0NBTEVfUkVUVVJOX05VTUJFUiwKICAgICAgICAgICAgICAgICAgICAoTFBXU1RSKSZvZW1fY3AsIHNpemVvZihvZW1fY3ApL3NpemVvZihXQ0hBUikgKTsKICAgIGlmICghdW5peF9jcCkKICAgICAgICBHZXRMb2NhbGVJbmZvVyggTE9DQUxFX1VTRVJfREVGQVVMVCwgTE9DQUxFX0lERUZBVUxUVU5JWENPREVQQUdFIHwgTE9DQUxFX1JFVFVSTl9OVU1CRVIsCiAgICAgICAgICAgICAgICAgICAgICAgIChMUFdTVFIpJnVuaXhfY3AsIHNpemVvZih1bml4X2NwKS9zaXplb2YoV0NIQVIpICk7CgogICAgaWYgKCEoYW5zaV9jcHRhYmxlID0gd2luZV9jcF9nZXRfdGFibGUoIGFuc2lfY3AgKSkpCiAgICAgICAgYW5zaV9jcHRhYmxlID0gd2luZV9jcF9nZXRfdGFibGUoIDEyNTIgKTsKICAgIGlmICghKG9lbV9jcHRhYmxlID0gd2luZV9jcF9nZXRfdGFibGUoIG9lbV9jcCApKSkKICAgICAgICBvZW1fY3B0YWJsZSAgPSB3aW5lX2NwX2dldF90YWJsZSggNDM3ICk7CiAgICBpZiAoIShtYWNfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBtYWNfY3AgKSkpCiAgICAgICAgbWFjX2NwdGFibGUgID0gd2luZV9jcF9nZXRfdGFibGUoIDEwMDAwICk7CiAgICBpZiAodW5peF9jcCAhPSBDUF9VVEY4KQogICAgewogICAgICAgIGlmICghKHVuaXhfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCB1bml4X2NwICkpKQogICAgICAgICAgICB1bml4X2NwdGFibGUgID0gd2luZV9jcF9nZXRfdGFibGUoIDI4NTkxICk7CiAgICB9CgogICAgX193aW5lX2luaXRfY29kZXBhZ2VzKCBhbnNpX2NwdGFibGUsIG9lbV9jcHRhYmxlLCB1bml4X2NwdGFibGUgKTsKCiAgICBUUkFDRSggImFuc2k9JTAzZCBvZW09JTAzZCBtYWM9JTAzZCB1bml4PSUwM2RcbiIsCiAgICAgICAgICAgYW5zaV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlLCBvZW1fY3B0YWJsZS0+aW5mby5jb2RlcGFnZSwKICAgICAgICAgICBtYWNfY3B0YWJsZS0+aW5mby5jb2RlcGFnZSwgdW5peF9jcCApOwoKICAgIHNldGxvY2FsZShMQ19OVU1FUklDLCAiQyIpOyAgLyogRklYTUU6IG9sZWF1dDMyIGRlcGVuZHMgb24gdGhpcyAqLwp9CgpzdGF0aWMgSEFORExFIE5MU19SZWdPcGVuS2V5KEhBTkRMRSBoUm9vdEtleSwgTFBDV1NUUiBzektleU5hbWUpCnsKICAgIFVOSUNPREVfU1RSSU5HIGtleU5hbWU7CiAgICBPQkpFQ1RfQVRUUklCVVRFUyBhdHRyOwogICAgSEFORExFIGhrZXk7CgogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZrZXlOYW1lLCBzektleU5hbWUgKTsKICAgIEluaXRpYWxpemVPYmplY3RBdHRyaWJ1dGVzKCZhdHRyLCAma2V5TmFtZSwgMCwgaFJvb3RLZXksIE5VTEwpOwoKICAgIGlmIChOdE9wZW5LZXkoICZoa2V5LCBLRVlfQUxMX0FDQ0VTUywgJmF0dHIgKSAhPSBTVEFUVVNfU1VDQ0VTUykKICAgICAgICBoa2V5ID0gMDsKCiAgICByZXR1cm4gaGtleTsKfQoKc3RhdGljIEJPT0wgTkxTX1JlZ0VudW1TdWJLZXkoSEFORExFIGhLZXksIFVJTlQgdWxJbmRleCwgTFBXU1RSIHN6S2V5TmFtZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUxPTkcga2V5TmFtZVNpemUpCnsKICAgIEJZVEUgYnVmZmVyWzgwXTsKICAgIEtFWV9CQVNJQ19JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfQkFTSUNfSU5GT1JNQVRJT04gKilidWZmZXI7CiAgICBEV09SRCBkd0xlbjsKCiAgICBpZiAoTnRFbnVtZXJhdGVLZXkoIGhLZXksIHVsSW5kZXgsIEtleUJhc2ljSW5mb3JtYXRpb24sIGJ1ZmZlciwKICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGJ1ZmZlciksICZkd0xlbikgIT0gU1RBVFVTX1NVQ0NFU1MgfHwKICAgICAgICBpbmZvLT5OYW1lTGVuZ3RoID4ga2V5TmFtZVNpemUpCiAgICB7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIFRSQUNFKCJpbmZvLT5OYW1lICVzIGluZm8tPk5hbWVMZW5ndGggJWRcbiIsIGRlYnVnc3RyX3coaW5mby0+TmFtZSksIGluZm8tPk5hbWVMZW5ndGgpOwoKICAgIG1lbWNweSggc3pLZXlOYW1lLCBpbmZvLT5OYW1lLCBpbmZvLT5OYW1lTGVuZ3RoKTsKICAgIHN6S2V5TmFtZVtpbmZvLT5OYW1lTGVuZ3RoIC8gc2l6ZW9mKFdDSEFSKV0gPSAnXDAnOwoKICAgIFRSQUNFKCJyZXR1cm5pbmcgJXNcbiIsIGRlYnVnc3RyX3coc3pLZXlOYW1lKSk7CiAgICByZXR1cm4gVFJVRTsKfQoKc3RhdGljIEJPT0wgTkxTX1JlZ0VudW1WYWx1ZShIQU5ETEUgaEtleSwgVUlOVCB1bEluZGV4LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQV1NUUiBzelZhbHVlTmFtZSwgVUxPTkcgdmFsdWVOYW1lU2l6ZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUFdTVFIgc3pWYWx1ZURhdGEsIFVMT05HIHZhbHVlRGF0YVNpemUpCnsKICAgIEJZVEUgYnVmZmVyWzgwXTsKICAgIEtFWV9WQUxVRV9GVUxMX0lORk9STUFUSU9OICppbmZvID0gKEtFWV9WQUxVRV9GVUxMX0lORk9STUFUSU9OICopYnVmZmVyOwogICAgRFdPUkQgZHdMZW47CgogICAgaWYgKE50RW51bWVyYXRlVmFsdWVLZXkoIGhLZXksIHVsSW5kZXgsIEtleVZhbHVlRnVsbEluZm9ybWF0aW9uLAogICAgICAgIGJ1ZmZlciwgc2l6ZW9mKGJ1ZmZlciksICZkd0xlbiApICE9IFNUQVRVU19TVUNDRVNTIHx8CiAgICAgICAgaW5mby0+TmFtZUxlbmd0aCA+IHZhbHVlTmFtZVNpemUgfHwKICAgICAgICBpbmZvLT5EYXRhTGVuZ3RoID4gdmFsdWVEYXRhU2l6ZSkKICAgIHsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgVFJBQ0UoImluZm8tPk5hbWUgJXMgaW5mby0+RGF0YUxlbmd0aCAlZFxuIiwgZGVidWdzdHJfdyhpbmZvLT5OYW1lKSwgaW5mby0+RGF0YUxlbmd0aCk7CgogICAgbWVtY3B5KCBzelZhbHVlTmFtZSwgaW5mby0+TmFtZSwgaW5mby0+TmFtZUxlbmd0aCk7CiAgICBzelZhbHVlTmFtZVtpbmZvLT5OYW1lTGVuZ3RoIC8gc2l6ZW9mKFdDSEFSKV0gPSAnXDAnOwogICAgbWVtY3B5KCBzelZhbHVlRGF0YSwgYnVmZmVyICsgaW5mby0+RGF0YU9mZnNldCwgaW5mby0+RGF0YUxlbmd0aCApOwogICAgc3pWYWx1ZURhdGFbaW5mby0+RGF0YUxlbmd0aCAvIHNpemVvZihXQ0hBUildID0gJ1wwJzsKCiAgICBUUkFDRSgicmV0dXJuaW5nICVzICVzXG4iLCBkZWJ1Z3N0cl93KHN6VmFsdWVOYW1lKSwgZGVidWdzdHJfdyhzelZhbHVlRGF0YSkpOwogICAgcmV0dXJuIFRSVUU7Cn0KCnN0YXRpYyBCT09MIE5MU19SZWdHZXREd29yZChIQU5ETEUgaEtleSwgTFBDV1NUUiBzelZhbHVlTmFtZSwgRFdPUkQgKmxwVmFsKQp7CiAgICBCWVRFIGJ1ZmZlclsxMjhdOwogICAgY29uc3QgS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKmluZm8gPSAoS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKilidWZmZXI7CiAgICBEV09SRCBkd1NpemUgPSBzaXplb2YoYnVmZmVyKTsKICAgIFVOSUNPREVfU1RSSU5HIHZhbHVlTmFtZTsKCiAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJnZhbHVlTmFtZSwgc3pWYWx1ZU5hbWUgKTsKCiAgICBUUkFDRSgiJXAsICVzXG4iLCBoS2V5LCBkZWJ1Z3N0cl93KHN6VmFsdWVOYW1lKSk7CiAgICBpZiAoTnRRdWVyeVZhbHVlS2V5KCBoS2V5LCAmdmFsdWVOYW1lLCBLZXlWYWx1ZVBhcnRpYWxJbmZvcm1hdGlvbiwKICAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZmZlciwgZHdTaXplLCAmZHdTaXplICkgPT0gU1RBVFVTX1NVQ0NFU1MgJiYKICAgICAgICBpbmZvLT5EYXRhTGVuZ3RoID09IHNpemVvZihEV09SRCkpCiAgICB7CiAgICAgICAgbWVtY3B5KGxwVmFsLCBpbmZvLT5EYXRhLCBzaXplb2YoRFdPUkQpKTsKICAgICAgICByZXR1cm4gVFJVRTsKICAgIH0KCiAgICByZXR1cm4gRkFMU0U7Cn0KCnN0YXRpYyBCT09MIE5MU19HZXRMYW5ndWFnZUdyb3VwTmFtZShMR1JQSUQgbGdycGlkLCBMUFdTVFIgc3pOYW1lLCBVTE9ORyBuYW1lU2l6ZSkKewogICAgTEFOR0lEICBsYW5nSWQ7CiAgICBMUENXU1RSIHN6UmVzb3VyY2VOYW1lID0gTUFLRUlOVFJFU09VUkNFVygoKGxncnBpZCArIDB4MjAwMCkgPj4gNCkgKyAxKTsKICAgIEhSU1JDICAgaFJlc291cmNlOwogICAgQk9PTCAgICBiUmV0ID0gRkFMU0U7CgogICAgLyogRklYTUU6IElzIGl0IGNvcnJlY3QgdG8gdXNlIHRoZSBzeXN0ZW0gZGVmYXVsdCBsYW5naWQ/ICovCiAgICBsYW5nSWQgPSBHZXRTeXN0ZW1EZWZhdWx0TGFuZ0lEKCk7CgogICAgaWYgKFNVQkxBTkdJRChsYW5nSWQpID09IFNVQkxBTkdfTkVVVFJBTCkKICAgICAgICBsYW5nSWQgPSBNQUtFTEFOR0lEKCBQUklNQVJZTEFOR0lEKGxhbmdJZCksIFNVQkxBTkdfREVGQVVMVCApOwoKICAgIGhSZXNvdXJjZSA9IEZpbmRSZXNvdXJjZUV4Vygga2VybmVsMzJfaGFuZGxlLCAoTFBXU1RSKVJUX1NUUklORywgc3pSZXNvdXJjZU5hbWUsIGxhbmdJZCApOwoKICAgIGlmIChoUmVzb3VyY2UpCiAgICB7CiAgICAgICAgSEdMT0JBTCBoUmVzRGlyID0gTG9hZFJlc291cmNlKCBrZXJuZWwzMl9oYW5kbGUsIGhSZXNvdXJjZSApOwoKICAgICAgICBpZiAoaFJlc0RpcikKICAgICAgICB7CiAgICAgICAgICAgIFVMT05HICAgaVJlc291cmNlSW5kZXggPSBsZ3JwaWQgJiAweGY7CiAgICAgICAgICAgIExQQ1dTVFIgbHBSZXNFbnRyeSA9IExvY2tSZXNvdXJjZSggaFJlc0RpciApOwogICAgICAgICAgICBVTE9ORyAgIGk7CgogICAgICAgICAgICBmb3IgKGkgPSAwOyBpIDwgaVJlc291cmNlSW5kZXg7IGkrKykKICAgICAgICAgICAgICAgIGxwUmVzRW50cnkgKz0gKmxwUmVzRW50cnkgKyAxOwoKICAgICAgICAgICAgaWYgKCpscFJlc0VudHJ5IDwgbmFtZVNpemUpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIG1lbWNweSggc3pOYW1lLCBscFJlc0VudHJ5ICsgMSwgKmxwUmVzRW50cnkgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgICAgICAgICBzek5hbWVbKmxwUmVzRW50cnldID0gJ1wwJzsKICAgICAgICAgICAgICAgIGJSZXQgPSBUUlVFOwogICAgICAgICAgICB9CgogICAgICAgIH0KICAgICAgICBGcmVlUmVzb3VyY2UoIGhSZXNvdXJjZSApOwogICAgfQogICAgcmV0dXJuIGJSZXQ7Cn0KCi8qIFJlZ2lzdHJ5IGtleXMgZm9yIE5MUyByZWxhdGVkIGluZm9ybWF0aW9uICovCgpzdGF0aWMgY29uc3QgV0NIQVIgc3pDb3VudHJ5TGlzdE5hbWVbXSA9IHsKICAgICdNJywnYScsJ2MnLCdoJywnaScsJ24nLCdlJywnXFwnLCdTJywnbycsJ2YnLCd0JywndycsJ2EnLCdyJywnZScsJ1xcJywKICAgICdNJywnaScsJ2MnLCdyJywnbycsJ3MnLCdvJywnZicsJ3QnLCdcXCcsJ1cnLCdpJywnbicsJ2QnLCdvJywndycsJ3MnLCdcXCcsCiAgICAnQycsJ3UnLCdyJywncicsJ2UnLCduJywndCcsJ1YnLCdlJywncicsJ3MnLCdpJywnbycsJ24nLCdcXCcsCiAgICAnVCcsJ2UnLCdsJywnZScsJ3AnLCdoJywnbycsJ24nLCd5JywnXFwnLAogICAgJ0MnLCdvJywndScsJ24nLCd0JywncicsJ3knLCcgJywnTCcsJ2knLCdzJywndCcsJ1wwJwp9OwoKCi8qIENhbGxiYWNrIGZ1bmN0aW9uIHB0cnMgZm9yIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc0EvVyAqLwp0eXBlZGVmIHN0cnVjdAp7CiAgTEFOR1VBR0VHUk9VUF9FTlVNUFJPQ0EgcHJvY0E7CiAgTEFOR1VBR0VHUk9VUF9FTlVNUFJPQ1cgcHJvY1c7CiAgRFdPUkQgICAgZHdGbGFnczsKICBMT05HX1BUUiBsUGFyYW07Cn0gRU5VTUxBTkdVQUdFR1JPVVBfQ0FMTEJBQ0tTOwoKLyogSW50ZXJuYWwgaW1wbGVtZW50YXRpb24gb2YgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzQS9XICovCnN0YXRpYyBCT09MIE5MU19FbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHMoRU5VTUxBTkdVQUdFR1JPVVBfQ0FMTEJBQ0tTICpscFByb2NzKQp7CiAgICBXQ0hBUiBzek51bWJlclsxMF0sIHN6VmFsdWVbNF07CiAgICBIQU5ETEUgaEtleTsKICAgIEJPT0wgYkNvbnRpbnVlID0gVFJVRTsKICAgIFVMT05HIHVsSW5kZXggPSAwOwoKICAgIGlmICghbHBQcm9jcykKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBzd2l0Y2ggKGxwUHJvY3MtPmR3RmxhZ3MpCiAgICB7CiAgICBjYXNlIDA6CiAgICAgICAgLyogRGVmYXVsdCB0byBMR1JQSURfSU5TVEFMTEVEICovCiAgICAgICAgbHBQcm9jcy0+ZHdGbGFncyA9IExHUlBJRF9JTlNUQUxMRUQ7CiAgICAgICAgLyogRmFsbCB0aHJvdWdoLi4uICovCiAgICBjYXNlIExHUlBJRF9JTlNUQUxMRUQ6CiAgICBjYXNlIExHUlBJRF9TVVBQT1JURUQ6CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaEtleSA9IE5MU19SZWdPcGVuS2V5KCAwLCBzekxhbmdHcm91cHNLZXlOYW1lICk7CgogICAgaWYgKCFoS2V5KQogICAgICAgIEZJWE1FKCJOTFMgcmVnaXN0cnkga2V5IG5vdCBmb3VuZC4gUGxlYXNlIGFwcGx5IHRoZSBkZWZhdWx0IHJlZ2lzdHJ5IGZpbGUgJ3dpbmUuaW5mJ1xuIik7CgogICAgd2hpbGUgKGJDb250aW51ZSkKICAgIHsKICAgICAgICBpZiAoTkxTX1JlZ0VudW1WYWx1ZSggaEtleSwgdWxJbmRleCwgc3pOdW1iZXIsIHNpemVvZihzek51bWJlciksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN6VmFsdWUsIHNpemVvZihzelZhbHVlKSApKQogICAgICAgIHsKICAgICAgICAgICAgQk9PTCBiSW5zdGFsbGVkID0gc3pWYWx1ZVswXSA9PSAnMScgPyBUUlVFIDogRkFMU0U7CiAgICAgICAgICAgIExHUlBJRCBsZ3JwaWQgPSBzdHJ0b3VsVyggc3pOdW1iZXIsIE5VTEwsIDE2ICk7CgogICAgICAgICAgICBUUkFDRSgiZ3JwaWQgJXMgKCVzaW5zdGFsbGVkKVxuIiwgZGVidWdzdHJfdyhzek51bWJlciksCiAgICAgICAgICAgICAgICAgICBiSW5zdGFsbGVkID8gIiIgOiAibm90ICIpOwoKICAgICAgICAgICAgaWYgKGxwUHJvY3MtPmR3RmxhZ3MgPT0gTEdSUElEX1NVUFBPUlRFRCB8fCBiSW5zdGFsbGVkKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBXQ0hBUiBzekdycE5hbWVbNDhdOwoKICAgICAgICAgICAgICAgIGlmICghTkxTX0dldExhbmd1YWdlR3JvdXBOYW1lKCBsZ3JwaWQsIHN6R3JwTmFtZSwgc2l6ZW9mKHN6R3JwTmFtZSkgLyBzaXplb2YoV0NIQVIpICkpCiAgICAgICAgICAgICAgICAgICAgc3pHcnBOYW1lWzBdID0gJ1wwJzsKCiAgICAgICAgICAgICAgICBpZiAobHBQcm9jcy0+cHJvY1cpCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gbHBQcm9jcy0+cHJvY1coIGxncnBpZCwgc3pOdW1iZXIsIHN6R3JwTmFtZSwgbHBQcm9jcy0+ZHdGbGFncywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbHBQcm9jcy0+bFBhcmFtICk7CiAgICAgICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgY2hhciBzek51bWJlckFbc2l6ZW9mKHN6TnVtYmVyKS9zaXplb2YoV0NIQVIpXTsKICAgICAgICAgICAgICAgICAgICBjaGFyIHN6R3JwTmFtZUFbNDhdOwoKICAgICAgICAgICAgICAgICAgICAvKiBGSVhNRTogTVNETiBkb2Vzbid0IHNheSB3aGljaCBjb2RlIHBhZ2UgdGhlIFctPkEgdHJhbnNsYXRpb24gdXNlcywKICAgICAgICAgICAgICAgICAgICAgKiAgICAgICAgb3Igd2hldGhlciB0aGUgbGFuZ3VhZ2UgbmFtZXMgYXJlIGV2ZXIgbG9jYWxpc2VkLiBBc3N1bWUgQ1BfQUNQLgogICAgICAgICAgICAgICAgICAgICAqLwoKICAgICAgICAgICAgICAgICAgICBXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgc3pOdW1iZXIsIC0xLCBzek51bWJlckEsIHNpemVvZihzek51bWJlckEpLCAwLCAwKTsKICAgICAgICAgICAgICAgICAgICBXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgc3pHcnBOYW1lLCAtMSwgc3pHcnBOYW1lQSwgc2l6ZW9mKHN6R3JwTmFtZUEpLCAwLCAwKTsKCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gbHBQcm9jcy0+cHJvY0EoIGxncnBpZCwgc3pOdW1iZXJBLCBzekdycE5hbWVBLCBscFByb2NzLT5kd0ZsYWdzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBscFByb2NzLT5sUGFyYW0gKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQoKICAgICAgICAgICAgdWxJbmRleCsrOwogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgICAgIGJDb250aW51ZSA9IEZBTFNFOwoKICAgICAgICBpZiAoIWJDb250aW51ZSkKICAgICAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKGhLZXkpCiAgICAgICAgTnRDbG9zZSggaEtleSApOwoKICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBICAgIChLRVJORUwzMi5AKQogKgogKiBDYWxsIGEgdXNlcnMgZnVuY3Rpb24gZm9yIGVhY2ggbGFuZ3VhZ2UgZ3JvdXAgYXZhaWxhYmxlIG9uIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgcExhbmdHcnBFbnVtUHJvYyBbSV0gQ2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCBmb3IgZWFjaCBsYW5ndWFnZSBncm91cAogKiAgZHdGbGFncyAgICAgICAgICBbSV0gTEdSUElEX1NVUFBPUlRFRD1BbGwgU3VwcG9ydGVkLCBMR1JQSURfSU5TVEFMTEVEPUluc3RhbGxlZCBvbmx5CiAqICBsUGFyYW0gICAgICAgICAgIFtJXSBVc2VyIHBhcmFtZXRlciB0byBwYXNzIHRvIHBMYW5nR3JwRW51bVByb2MKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc0EoTEFOR1VBR0VHUk9VUF9FTlVNUFJPQ0EgcExhbmdHcnBFbnVtUHJvYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1MQU5HVUFHRUdST1VQX0NBTExCQUNLUyBwcm9jczsKCiAgICBUUkFDRSgiKCVwLDB4JTA4WCwweCUwOGxYKVxuIiwgcExhbmdHcnBFbnVtUHJvYywgZHdGbGFncywgbFBhcmFtKTsKCiAgICBwcm9jcy5wcm9jQSA9IHBMYW5nR3JwRW51bVByb2M7CiAgICBwcm9jcy5wcm9jVyA9IE5VTEw7CiAgICBwcm9jcy5kd0ZsYWdzID0gZHdGbGFnczsKICAgIHByb2NzLmxQYXJhbSA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1TeXN0ZW1MYW5ndWFnZUdyb3VwcyggcExhbmdHcnBFbnVtUHJvYyA/ICZwcm9jcyA6IE5VTEwpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzQS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc1coTEFOR1VBR0VHUk9VUF9FTlVNUFJPQ1cgcExhbmdHcnBFbnVtUHJvYywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1MQU5HVUFHRUdST1VQX0NBTExCQUNLUyBwcm9jczsKCiAgICBUUkFDRSgiKCVwLDB4JTA4WCwweCUwOGxYKVxuIiwgcExhbmdHcnBFbnVtUHJvYywgZHdGbGFncywgbFBhcmFtKTsKCiAgICBwcm9jcy5wcm9jQSA9IE5VTEw7CiAgICBwcm9jcy5wcm9jVyA9IHBMYW5nR3JwRW51bVByb2M7CiAgICBwcm9jcy5kd0ZsYWdzID0gZHdGbGFnczsKICAgIHByb2NzLmxQYXJhbSA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1TeXN0ZW1MYW5ndWFnZUdyb3VwcyggcExhbmdHcnBFbnVtUHJvYyA/ICZwcm9jcyA6IE5VTEwpOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc1ZhbGlkTGFuZ3VhZ2VHcm91cCAgICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgbGFuZ3VhZ2UgZ3JvdXAgaXMgc3VwcG9ydGVkIGFuZC9vciBpbnN0YWxsZWQuCiAqCiAqIFBBUkFNUwogKiAgbGdycGlkICBbSV0gTGFuZ3VhZ2UgR3JvdXAgSWQgKExHUlBJRF8gdmFsdWVzIGZyb20gIndpbm5scy5oIikKICogIGR3RmxhZ3MgW0ldIExHUlBJRF9TVVBQT1JURUQ9U3VwcG9ydGVkLCBMR1JQSURfSU5TVEFMTEVEPUluc3RhbGxlZAogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBpZiBsZ3JwaWQgaXMgc3VwcG9ydGVkIGFuZC9vciBpbnN0YWxsZWQsIGFjY29yZGluZyB0byBkd0ZsYWdzLgogKiAgRkFMU0Ugb3RoZXJ3aXNlLgogKi8KQk9PTCBXSU5BUEkgSXNWYWxpZExhbmd1YWdlR3JvdXAoTEdSUElEIGxncnBpZCwgRFdPUkQgZHdGbGFncykKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIHN6Rm9ybWF0W10gPSB7ICclJywneCcsJ1wwJyB9OwogICAgV0NIQVIgc3pWYWx1ZU5hbWVbMTZdLCBzelZhbHVlWzJdOwogICAgQk9PTCBiU3VwcG9ydGVkID0gRkFMU0UsIGJJbnN0YWxsZWQgPSBGQUxTRTsKICAgIEhBTkRMRSBoS2V5OwoKCiAgICBzd2l0Y2ggKGR3RmxhZ3MpCiAgICB7CiAgICBjYXNlIExHUlBJRF9JTlNUQUxMRUQ6CiAgICBjYXNlIExHUlBJRF9TVVBQT1JURUQ6CgogICAgICAgIGhLZXkgPSBOTFNfUmVnT3BlbktleSggMCwgc3pMYW5nR3JvdXBzS2V5TmFtZSApOwoKICAgICAgICBzcHJpbnRmVyggc3pWYWx1ZU5hbWUsIHN6Rm9ybWF0LCBsZ3JwaWQgKTsKCiAgICAgICAgaWYgKE5MU19SZWdHZXREd29yZCggaEtleSwgc3pWYWx1ZU5hbWUsIChMUERXT1JEKSZzelZhbHVlICkpCiAgICAgICAgewogICAgICAgICAgICBiU3VwcG9ydGVkID0gVFJVRTsKCiAgICAgICAgICAgIGlmIChzelZhbHVlWzBdID09ICcxJykKICAgICAgICAgICAgICAgIGJJbnN0YWxsZWQgPSBUUlVFOwogICAgICAgIH0KCiAgICAgICAgaWYgKGhLZXkpCiAgICAgICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKChkd0ZsYWdzID09IExHUlBJRF9TVVBQT1JURUQgJiYgYlN1cHBvcnRlZCkgfHwKICAgICAgICAoZHdGbGFncyA9PSBMR1JQSURfSU5TVEFMTEVEICYmIGJJbnN0YWxsZWQpKQogICAgICAgIHJldHVybiBUUlVFOwoKICAgIHJldHVybiBGQUxTRTsKfQoKLyogQ2FsbGJhY2sgZnVuY3Rpb24gcHRycyBmb3IgRW51bUxhbmd1YWdlR3JvdXBsb2NhbGVzQS9XICovCnR5cGVkZWYgc3RydWN0CnsKICBMQU5HR1JPVVBMT0NBTEVfRU5VTVBST0NBIHByb2NBOwogIExBTkdHUk9VUExPQ0FMRV9FTlVNUFJPQ1cgcHJvY1c7CiAgRFdPUkQgICAgZHdGbGFnczsKICBMR1JQSUQgICBsZ3JwaWQ7CiAgTE9OR19QVFIgbFBhcmFtOwp9IEVOVU1MQU5HVUFHRUdST1VQTE9DQUxFX0NBTExCQUNLUzsKCi8qIEludGVybmFsIGltcGxlbWVudGF0aW9uIG9mIEVudW1MYW5ndWFnZUdyb3VwbG9jYWxlc0EvVyAqLwpzdGF0aWMgQk9PTCBOTFNfRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzKEVOVU1MQU5HVUFHRUdST1VQTE9DQUxFX0NBTExCQUNLUyAqbHBQcm9jcykKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIHN6QWx0ZXJuYXRlU29ydHNLZXlOYW1lW10gPSB7CiAgICAgICdBJywnbCcsJ3QnLCdlJywncicsJ24nLCdhJywndCcsJ2UnLCcgJywnUycsJ28nLCdyJywndCcsJ3MnLCdcMCcKICAgIH07CiAgICBXQ0hBUiBzek51bWJlclsxMF0sIHN6VmFsdWVbNF07CiAgICBIQU5ETEUgaEtleTsKICAgIEJPT0wgYkNvbnRpbnVlID0gVFJVRSwgYkFsdGVybmF0ZSA9IEZBTFNFOwogICAgTEdSUElEIGxncnBpZDsKICAgIFVMT05HIHVsSW5kZXggPSAxOyAgLyogSWdub3JlIGRlZmF1bHQgZW50cnkgb2YgMXN0IGtleSAqLwoKICAgIGlmICghbHBQcm9jcyB8fCAhbHBQcm9jcy0+bGdycGlkIHx8IGxwUHJvY3MtPmxncnBpZCA+IExHUlBJRF9BUk1FTklBTikKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAobHBQcm9jcy0+ZHdGbGFncykKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGhLZXkgPSBOTFNfUmVnT3BlbktleSggMCwgc3pMb2NhbGVLZXlOYW1lICk7CgogICAgaWYgKCFoS2V5KQogICAgICAgIFdBUk4oIk5MUyByZWdpc3RyeSBrZXkgbm90IGZvdW5kLiBQbGVhc2UgYXBwbHkgdGhlIGRlZmF1bHQgcmVnaXN0cnkgZmlsZSAnd2luZS5pbmYnXG4iKTsKCiAgICB3aGlsZSAoYkNvbnRpbnVlKQogICAgewogICAgICAgIGlmIChOTFNfUmVnRW51bVZhbHVlKCBoS2V5LCB1bEluZGV4LCBzek51bWJlciwgc2l6ZW9mKHN6TnVtYmVyKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc3pWYWx1ZSwgc2l6ZW9mKHN6VmFsdWUpICkpCiAgICAgICAgewogICAgICAgICAgICBsZ3JwaWQgPSBzdHJ0b3VsVyggc3pWYWx1ZSwgTlVMTCwgMTYgKTsKCiAgICAgICAgICAgIFRSQUNFKCJsY2lkICVzLCBncnBpZCAlZCAoJXNtYXRjaGVkKVxuIiwgZGVidWdzdHJfdyhzek51bWJlciksCiAgICAgICAgICAgICAgICAgICBsZ3JwaWQsIGxncnBpZCA9PSBscFByb2NzLT5sZ3JwaWQgPyAiIiA6ICJub3QgIik7CgogICAgICAgICAgICBpZiAobGdycGlkID09IGxwUHJvY3MtPmxncnBpZCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgTENJRCBsY2lkOwoKICAgICAgICAgICAgICAgIGxjaWQgPSBzdHJ0b3VsVyggc3pOdW1iZXIsIE5VTEwsIDE2ICk7CgogICAgICAgICAgICAgICAgLyogRklYTUU6IG5hdGl2ZSByZXR1cm5zIGV4dHJhIHRleHQgZm9yIGEgZmV3ICgxNy8xNTApIGxvY2FsZXMsIGUuZzoKICAgICAgICAgICAgICAgICAqICcwMDAwMDQzNyAgICAgICAgICA7R2VvcmdpYW4nCiAgICAgICAgICAgICAgICAgKiBBdCBwcmVzZW50IHdlIG9ubHkgcGFzcyB0aGUgTENJRCBzdHJpbmcuCiAgICAgICAgICAgICAgICAgKi8KCiAgICAgICAgICAgICAgICBpZiAobHBQcm9jcy0+cHJvY1cpCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gbHBQcm9jcy0+cHJvY1coIGxncnBpZCwgbGNpZCwgc3pOdW1iZXIsIGxwUHJvY3MtPmxQYXJhbSApOwogICAgICAgICAgICAgICAgZWxzZQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgIGNoYXIgc3pOdW1iZXJBW3NpemVvZihzek51bWJlcikvc2l6ZW9mKFdDSEFSKV07CgogICAgICAgICAgICAgICAgICAgIFdpZGVDaGFyVG9NdWx0aUJ5dGUoQ1BfQUNQLCAwLCBzek51bWJlciwgLTEsIHN6TnVtYmVyQSwgc2l6ZW9mKHN6TnVtYmVyQSksIDAsIDApOwoKICAgICAgICAgICAgICAgICAgICBiQ29udGludWUgPSBscFByb2NzLT5wcm9jQSggbGdycGlkLCBsY2lkLCBzek51bWJlckEsIGxwUHJvY3MtPmxQYXJhbSApOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CgogICAgICAgICAgICB1bEluZGV4Kys7CiAgICAgICAgfQogICAgICAgIGVsc2UKICAgICAgICB7CiAgICAgICAgICAgIC8qIEZpbmlzaGVkIGVudW1lcmF0aW5nIHRoaXMga2V5ICovCiAgICAgICAgICAgIGlmICghYkFsdGVybmF0ZSkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgLyogRW51bWVyYXRlIGFsdGVybmF0ZSBzb3J0cyBhbHNvICovCiAgICAgICAgICAgICAgICBoS2V5ID0gTkxTX1JlZ09wZW5LZXkoIGhLZXksIHN6QWx0ZXJuYXRlU29ydHNLZXlOYW1lICk7CiAgICAgICAgICAgICAgICBiQWx0ZXJuYXRlID0gVFJVRTsKICAgICAgICAgICAgICAgIHVsSW5kZXggPSAwOwogICAgICAgICAgICB9CiAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IEZBTFNFOyAvKiBGaW5pc2hlZCBib3RoIGtleXMgKi8KICAgICAgICB9CgogICAgICAgIGlmICghYkNvbnRpbnVlKQogICAgICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAoaEtleSkKICAgICAgICBOdENsb3NlKCBoS2V5ICk7CgogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEVudW1MYW5ndWFnZUdyb3VwTG9jYWxlc0EgICAgKEtFUk5FTDMyLkApCiAqCiAqIENhbGwgYSB1c2VycyBmdW5jdGlvbiBmb3IgZXZlcnkgbG9jYWxlIGluIGEgbGFuZ3VhZ2UgZ3JvdXAgYXZhaWxhYmxlIG9uIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgcExhbmdHcnBMY0VudW1Qcm9jIFtJXSBDYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIGZvciBlYWNoIGxvY2FsZQogKiAgbGdycGlkICAgICAgICAgICAgIFtJXSBMYW5ndWFnZSBncm91cCAoTEdSUElEXyB2YWx1ZXMgZnJvbSAid2lubmxzLmgiKQogKiAgZHdGbGFncyAgICAgICAgICAgIFtJXSBSZXNlcnZlZCwgc2V0IHRvIDAKICogIGxQYXJhbSAgICAgICAgICAgICBbSV0gVXNlciBwYXJhbWV0ZXIgdG8gcGFzcyB0byBwTGFuZ0dycExjRW51bVByb2MKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEVudW1MYW5ndWFnZUdyb3VwTG9jYWxlc0EoTEFOR0dST1VQTE9DQUxFX0VOVU1QUk9DQSBwTGFuZ0dycExjRW51bVByb2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTEdSUElEIGxncnBpZCwgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNTEFOR1VBR0VHUk9VUExPQ0FMRV9DQUxMQkFDS1MgY2FsbGJhY2tzOwoKICAgIFRSQUNFKCIoJXAsMHglMDhYLDB4JTA4WCwweCUwOGxYKVxuIiwgcExhbmdHcnBMY0VudW1Qcm9jLCBsZ3JwaWQsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgY2FsbGJhY2tzLnByb2NBICAgPSBwTGFuZ0dycExjRW51bVByb2M7CiAgICBjYWxsYmFja3MucHJvY1cgICA9IE5VTEw7CiAgICBjYWxsYmFja3MuZHdGbGFncyA9IGR3RmxhZ3M7CiAgICBjYWxsYmFja3MubGdycGlkICA9IGxncnBpZDsKICAgIGNhbGxiYWNrcy5sUGFyYW0gID0gbFBhcmFtOwoKICAgIHJldHVybiBOTFNfRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzKCBwTGFuZ0dycExjRW51bVByb2MgPyAmY2FsbGJhY2tzIDogTlVMTCApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXNXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzQS4KICovCkJPT0wgV0lOQVBJIEVudW1MYW5ndWFnZUdyb3VwTG9jYWxlc1coTEFOR0dST1VQTE9DQUxFX0VOVU1QUk9DVyBwTGFuZ0dycExjRW51bVByb2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTEdSUElEIGxncnBpZCwgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNTEFOR1VBR0VHUk9VUExPQ0FMRV9DQUxMQkFDS1MgY2FsbGJhY2tzOwoKICAgIFRSQUNFKCIoJXAsMHglMDhYLDB4JTA4WCwweCUwOGxYKVxuIiwgcExhbmdHcnBMY0VudW1Qcm9jLCBsZ3JwaWQsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgY2FsbGJhY2tzLnByb2NBICAgPSBOVUxMOwogICAgY2FsbGJhY2tzLnByb2NXICAgPSBwTGFuZ0dycExjRW51bVByb2M7CiAgICBjYWxsYmFja3MuZHdGbGFncyA9IGR3RmxhZ3M7CiAgICBjYWxsYmFja3MubGdycGlkICA9IGxncnBpZDsKICAgIGNhbGxiYWNrcy5sUGFyYW0gID0gbFBhcmFtOwoKICAgIHJldHVybiBOTFNfRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzKCBwTGFuZ0dycExjRW51bVByb2MgPyAmY2FsbGJhY2tzIDogTlVMTCApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtU3lzdGVtR2VvSUQgICAgKEtFUk5FTDMyLkApCiAqCiAqIENhbGwgYSB1c2VycyBmdW5jdGlvbiBmb3IgZXZlcnkgbG9jYXRpb24gYXZhaWxhYmxlIG9uIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgZ2VvY2xhc3MgICAgIFtJXSBUeXBlIG9mIGluZm9ybWF0aW9uIGRlc2lyZWQgKFNZU0dFT1RZUEUgZW51bSBmcm9tICJ3aW5ubHMuaCIpCiAqICByZXNlcnZlZCAgICAgW0ldIFJlc2VydmVkLCBzZXQgdG8gMAogKiAgcEdlb0VudW1Qcm9jIFtJXSBDYWxsYmFjayBmdW5jdGlvbiB0byBjYWxsIGZvciBlYWNoIGxvY2F0aW9uCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtR2VvSUQoR0VPQ0xBU1MgZ2VvY2xhc3MsIEdFT0lEIHJlc2VydmVkLCBHRU9fRU5VTVBST0MgcEdlb0VudW1Qcm9jKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc3pDb3VudHJ5Q29kZVZhbHVlTmFtZVtdID0gewogICAgICAnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsJ0MnLCdvJywnZCcsJ2UnLCdcMCcKICAgIH07CiAgICBXQ0hBUiBzek51bWJlclsxMF07CiAgICBIQU5ETEUgaEtleTsKICAgIFVMT05HIHVsSW5kZXggPSAwOwoKICAgIFRSQUNFKCIoMHglMDhYLDB4JTA4WCwlcClcbiIsIGdlb2NsYXNzLCByZXNlcnZlZCwgcEdlb0VudW1Qcm9jKTsKCiAgICBpZiAoZ2VvY2xhc3MgIT0gR0VPQ0xBU1NfTkFUSU9OIHx8IHJlc2VydmVkIHx8ICFwR2VvRW51bVByb2MpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaEtleSA9IE5MU19SZWdPcGVuS2V5KCAwLCBzekNvdW50cnlMaXN0TmFtZSApOwoKICAgIHdoaWxlIChOTFNfUmVnRW51bVN1YktleSggaEtleSwgdWxJbmRleCwgc3pOdW1iZXIsIHNpemVvZihzek51bWJlcikgKSkKICAgIHsKICAgICAgICBCT09MIGJDb250aW51ZSA9IFRSVUU7CiAgICAgICAgRFdPUkQgZHdHZW9JZDsKICAgICAgICBIQU5ETEUgaFN1YktleSA9IE5MU19SZWdPcGVuS2V5KCBoS2V5LCBzek51bWJlciApOwoKICAgICAgICBpZiAoaFN1YktleSkKICAgICAgICB7CiAgICAgICAgICAgIGlmIChOTFNfUmVnR2V0RHdvcmQoIGhTdWJLZXksIHN6Q291bnRyeUNvZGVWYWx1ZU5hbWUsICZkd0dlb0lkICkpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFRSQUNFKCJHb3QgZ2VvaWQgJWRcbiIsIGR3R2VvSWQpOwoKICAgICAgICAgICAgICAgIGlmICghcEdlb0VudW1Qcm9jKCBkd0dlb0lkICkpCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gRkFMU0U7CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIE50Q2xvc2UoIGhTdWJLZXkgKTsKICAgICAgICB9CgogICAgICAgIGlmICghYkNvbnRpbnVlKQogICAgICAgICAgICBicmVhazsKCiAgICAgICAgdWxJbmRleCsrOwogICAgfQoKICAgIGlmIChoS2V5KQogICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSW52YWxpZGF0ZU5MU0NhY2hlICAgICAgICAgICAoS0VSTkVMMzIuQCkKICoKICogSW52YWxpZGF0ZSB0aGUgY2FjaGUgb2YgTkxTIHZhbHVlcy4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLgogKiAgRmFpbHVyZTogRkFMU0UuCiAqLwpCT09MIFdJTkFQSSBJbnZhbGlkYXRlTkxTQ2FjaGUodm9pZCkKewogIEZJWE1FKCIoKSBzdHViXG4iKTsKICByZXR1cm4gRkFMU0U7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFVzZXJHZW9JRCAoS0VSTkVMMzIuQCkKICovCkdFT0lEIFdJTkFQSSBHZXRVc2VyR2VvSUQoIEdFT0NMQVNTIEdlb0NsYXNzICkKewogICAgR0VPSUQgcmV0ID0gR0VPSURfTk9UX0FWQUlMQUJMRTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBnZW9XW10gPSB7J0cnLCdlJywnbycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbmF0aW9uV1tdID0geydOJywnYScsJ3QnLCdpJywnbycsJ24nLDB9OwogICAgV0NIQVIgYnVmZmVyV1s0MF0sICplbmQ7CiAgICBEV09SRCBjb3VudDsKICAgIEhBTkRMRSBoa2V5LCBoU3Via2V5ID0gMDsKICAgIFVOSUNPREVfU1RSSU5HIGtleVc7CiAgICBjb25zdCBLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiAqKWJ1ZmZlclc7CiAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJmtleVcsIG5hdGlvblcgKTsKICAgIGNvdW50ID0gc2l6ZW9mKGJ1ZmZlclcpOwoKICAgIGlmKCEoaGtleSA9IGNyZWF0ZV9yZWdpc3RyeV9rZXkoKSkpIHJldHVybiByZXQ7CgogICAgc3dpdGNoKCBHZW9DbGFzcyApewogICAgY2FzZSBHRU9DTEFTU19OQVRJT046CiAgICAgICAgaWYgKChoU3Via2V5ID0gTkxTX1JlZ09wZW5LZXkoaGtleSwgZ2VvVykpKQogICAgICAgIHsKICAgICAgICAgICAgaWYoKE50UXVlcnlWYWx1ZUtleShoU3Via2V5LCAma2V5VywgS2V5VmFsdWVQYXJ0aWFsSW5mb3JtYXRpb24sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQQllURSlidWZmZXJXLCBjb3VudCwgJmNvdW50KSA9PSBTVEFUVVNfU1VDQ0VTUyApICYmIGluZm8tPkRhdGFMZW5ndGgpCiAgICAgICAgICAgICAgICByZXQgPSBzdHJ0b2xXKChMUENXU1RSKWluZm8tPkRhdGEsICZlbmQsIDEwKTsKICAgICAgICB9CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIEdFT0NMQVNTX1JFR0lPTjoKICAgICAgICBGSVhNRSgiR0VPQ0xBU1NfUkVHSU9OIG5vdCBoYW5kbGVkIHlldFxuIik7CiAgICAgICAgYnJlYWs7CiAgICB9CgogICAgTnRDbG9zZShoa2V5KTsKICAgIGlmIChoU3Via2V5KSBOdENsb3NlKGhTdWJrZXkpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgU2V0VXNlckdlb0lEIChLRVJORUwzMi5AKQogKi8KQk9PTCBXSU5BUEkgU2V0VXNlckdlb0lEKCBHRU9JRCBHZW9JRCApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBnZW9XW10gPSB7J0cnLCdlJywnbycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbmF0aW9uV1tdID0geydOJywnYScsJ3QnLCdpJywnbycsJ24nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGZvcm1hdFdbXSA9IHsnJScsJ2knLDB9OwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVcsa2V5VzsKICAgIFdDSEFSIGJ1ZmZlcldbMTBdOwogICAgT0JKRUNUX0FUVFJJQlVURVMgYXR0cjsKICAgIEhBTkRMRSBoa2V5OwoKICAgIGlmKCEoaGtleSA9IGNyZWF0ZV9yZWdpc3RyeV9rZXkoKSkpIHJldHVybiBGQUxTRTsKCiAgICBhdHRyLkxlbmd0aCA9IHNpemVvZihhdHRyKTsKICAgIGF0dHIuUm9vdERpcmVjdG9yeSA9IGhrZXk7CiAgICBhdHRyLk9iamVjdE5hbWUgPSAmbmFtZVc7CiAgICBhdHRyLkF0dHJpYnV0ZXMgPSAwOwogICAgYXR0ci5TZWN1cml0eURlc2NyaXB0b3IgPSBOVUxMOwogICAgYXR0ci5TZWN1cml0eVF1YWxpdHlPZlNlcnZpY2UgPSBOVUxMOwogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgZ2VvVyApOwogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZrZXlXLCBuYXRpb25XICk7CgogICAgaWYgKE50Q3JlYXRlS2V5KCAmaGtleSwgS0VZX0FMTF9BQ0NFU1MsICZhdHRyLCAwLCBOVUxMLCAwLCBOVUxMICkgIT0gU1RBVFVTX1NVQ0NFU1MpCgogICAgewogICAgICAgIE50Q2xvc2UoYXR0ci5Sb290RGlyZWN0b3J5KTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgc3ByaW50ZlcoYnVmZmVyVywgZm9ybWF0VywgR2VvSUQpOwogICAgTnRTZXRWYWx1ZUtleShoa2V5LCAma2V5VywgMCwgUkVHX1NaLCBidWZmZXJXLCAoc3RybGVuVyhidWZmZXJXKSArIDEpICogc2l6ZW9mKFdDSEFSKSk7CiAgICBOdENsb3NlKGF0dHIuUm9vdERpcmVjdG9yeSk7CiAgICBOdENsb3NlKGhrZXkpOwogICAgcmV0dXJuIFRSVUU7Cn0KCnR5cGVkZWYgc3RydWN0CnsKICAgIHVuaW9uCiAgICB7CiAgICAgICAgVUlMQU5HVUFHRV9FTlVNUFJPQ0EgcHJvY0E7CiAgICAgICAgVUlMQU5HVUFHRV9FTlVNUFJPQ1cgcHJvY1c7CiAgICB9IHU7CiAgICBEV09SRCBmbGFnczsKICAgIExPTkdfUFRSIHBhcmFtOwp9IEVOVU1fVUlMQU5HX0NBTExCQUNLOwoKc3RhdGljIEJPT0wgQ0FMTEJBQ0sgZW51bV91aWxhbmdfcHJvY19hKCBITU9EVUxFIGhNb2R1bGUsIExQQ1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1NUUiBuYW1lLCBXT1JEIExhbmdJRCwgTE9OR19QVFIgbFBhcmFtICkKewogICAgRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgKmVudW1fdWlsYW5nID0gKEVOVU1fVUlMQU5HX0NBTExCQUNLICopbFBhcmFtOwogICAgY2hhciBidWZbMjBdOwoKICAgIHNwcmludGYoYnVmLCAiJTA4eCIsIChVSU5UKUxhbmdJRCk7CiAgICByZXR1cm4gZW51bV91aWxhbmctPnUucHJvY0EoIGJ1ZiwgZW51bV91aWxhbmctPnBhcmFtICk7Cn0KCnN0YXRpYyBCT09MIENBTExCQUNLIGVudW1fdWlsYW5nX3Byb2NfdyggSE1PRFVMRSBoTW9kdWxlLCBMUENXU1RSIHR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDV1NUUiBuYW1lLCBXT1JEIExhbmdJRCwgTE9OR19QVFIgbFBhcmFtICkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGZvcm1hdFdbXSA9IHsnJScsJzAnLCc4JywneCcsMH07CiAgICBFTlVNX1VJTEFOR19DQUxMQkFDSyAqZW51bV91aWxhbmcgPSAoRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgKilsUGFyYW07CiAgICBXQ0hBUiBidWZbMjBdOwoKICAgIHNwcmludGZXKCBidWYsIGZvcm1hdFcsIChVSU5UKUxhbmdJRCApOwogICAgcmV0dXJuIGVudW1fdWlsYW5nLT51LnByb2NXKCBidWYsIGVudW1fdWlsYW5nLT5wYXJhbSApOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtVUlMYW5ndWFnZXNBIChLRVJORUwzMi5AKQogKi8KQk9PTCBXSU5BUEkgRW51bVVJTGFuZ3VhZ2VzQShVSUxBTkdVQUdFX0VOVU1QUk9DQSBwVUlMYW5nRW51bVByb2MsIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgZW51bV91aWxhbmc7CgogICAgVFJBQ0UoIiVwLCAleCwgJWx4XG4iLCBwVUlMYW5nRW51bVByb2MsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgaWYoIXBVSUxhbmdFbnVtUHJvYykgewoJU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKCXJldHVybiBGQUxTRTsKICAgIH0KICAgIGlmKGR3RmxhZ3MpIHsKCVNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBlbnVtX3VpbGFuZy51LnByb2NBID0gcFVJTGFuZ0VudW1Qcm9jOwogICAgZW51bV91aWxhbmcuZmxhZ3MgPSBkd0ZsYWdzOwogICAgZW51bV91aWxhbmcucGFyYW0gPSBsUGFyYW07CgogICAgRW51bVJlc291cmNlTGFuZ3VhZ2VzQSgga2VybmVsMzJfaGFuZGxlLCAoTFBDU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENTVFIpTE9DQUxFX0lMQU5HVUFHRSwgZW51bV91aWxhbmdfcHJvY19hLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExPTkdfUFRSKSZlbnVtX3VpbGFuZyk7CiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVVJTGFuZ3VhZ2VzVyAoS0VSTkVMMzIuQCkKICovCkJPT0wgV0lOQVBJIEVudW1VSUxhbmd1YWdlc1coVUlMQU5HVUFHRV9FTlVNUFJPQ1cgcFVJTGFuZ0VudW1Qcm9jLCBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1fVUlMQU5HX0NBTExCQUNLIGVudW1fdWlsYW5nOwoKICAgIFRSQUNFKCIlcCwgJXgsICVseFxuIiwgcFVJTGFuZ0VudW1Qcm9jLCBkd0ZsYWdzLCBsUGFyYW0pOwoKCiAgICBpZighcFVJTGFuZ0VudW1Qcm9jKSB7CglTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQogICAgaWYoZHdGbGFncykgewoJU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGVudW1fdWlsYW5nLnUucHJvY1cgPSBwVUlMYW5nRW51bVByb2M7CiAgICBlbnVtX3VpbGFuZy5mbGFncyA9IGR3RmxhZ3M7CiAgICBlbnVtX3VpbGFuZy5wYXJhbSA9IGxQYXJhbTsKCiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNXKCBrZXJuZWwzMl9oYW5kbGUsIChMUENXU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIGVudW1fdWlsYW5nX3Byb2NfdywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMT05HX1BUUikmZW51bV91aWxhbmcpOwogICAgcmV0dXJuIFRSVUU7Cn0KCklOVCBXSU5BUEkgR2V0R2VvSW5mb1coR0VPSUQgR2VvSWQsIEdFT1RZUEUgR2VvVHlwZSwgTFBXU1RSIGxwR2VvRGF0YSwgCiAgICAgICAgICAgICAgICBpbnQgY2NoRGF0YSwgTEFOR0lEIGxhbmd1YWdlKQp7CiAgICBGSVhNRSgiJWQgJWQgJXAgJWQgJWRcbiIsIEdlb0lkLCBHZW9UeXBlLCBscEdlb0RhdGEsIGNjaERhdGEsIGxhbmd1YWdlKTsKICAgIHJldHVybiAwOwp9CgpJTlQgV0lOQVBJIEdldEdlb0luZm9BKEdFT0lEIEdlb0lkLCBHRU9UWVBFIEdlb1R5cGUsIExQU1RSIGxwR2VvRGF0YSwgCiAgICAgICAgICAgICAgICBpbnQgY2NoRGF0YSwgTEFOR0lEIGxhbmd1YWdlKQp7CiAgICBGSVhNRSgiJWQgJWQgJXAgJWQgJWRcbiIsIEdlb0lkLCBHZW9UeXBlLCBscEdlb0RhdGEsIGNjaERhdGEsIGxhbmd1YWdlKTsKICAgIHJldHVybiAwOwp9Cg==