LyoKICogTG9jYWxlIHN1cHBvcnQKICoKICogQ29weXJpZ2h0IDE5OTUgTWFydGluIHZvbiBMb2V3aXMKICogQ29weXJpZ2h0IDE5OTggRGF2aWQgTGVlIExhbWJlcnQKICogQ29weXJpZ2h0IDIwMDAgSnVsaW8gQ+lzYXIgR+F6cXVlegogKiBDb3B5cmlnaHQgMjAwMiBBbGV4YW5kcmUgSnVsbGlhcmQgZm9yIENvZGVXZWF2ZXJzCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBmcmVlIHNvZnR3YXJlOyB5b3UgY2FuIHJlZGlzdHJpYnV0ZSBpdCBhbmQvb3IKICogbW9kaWZ5IGl0IHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgR05VIExlc3NlciBHZW5lcmFsIFB1YmxpYwogKiBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieSB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uOyBlaXRoZXIKICogdmVyc2lvbiAyLjEgb2YgdGhlIExpY2Vuc2UsIG9yIChhdCB5b3VyIG9wdGlvbikgYW55IGxhdGVyIHZlcnNpb24uCiAqCiAqIFRoaXMgbGlicmFyeSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogKiBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogKiBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlIEdOVQogKiBMZXNzZXIgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBmb3IgbW9yZSBkZXRhaWxzLgogKgogKiBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgTGVzc2VyIEdlbmVyYWwgUHVibGljCiAqIExpY2Vuc2UgYWxvbmcgd2l0aCB0aGlzIGxpYnJhcnk7IGlmIG5vdCwgd3JpdGUgdG8gdGhlIEZyZWUgU29mdHdhcmUKICogRm91bmRhdGlvbiwgSW5jLiwgNTEgRnJhbmtsaW4gU3QsIEZpZnRoIEZsb29yLCBCb3N0b24sIE1BIDAyMTEwLTEzMDEsIFVTQQogKi8KCiNpbmNsdWRlICJjb25maWcuaCIKI2luY2x1ZGUgIndpbmUvcG9ydC5oIgoKI2luY2x1ZGUgPGFzc2VydC5oPgojaW5jbHVkZSA8bG9jYWxlLmg+CiNpbmNsdWRlIDxzdHJpbmcuaD4KI2luY2x1ZGUgPHN0ZGFyZy5oPgojaW5jbHVkZSA8c3RkaW8uaD4KI2luY2x1ZGUgPGN0eXBlLmg+CiNpbmNsdWRlIDxzdGRsaWIuaD4KCiNpZmRlZiBfX0FQUExFX18KIyBpbmNsdWRlIDxDb3JlRm91bmRhdGlvbi9DRkJ1bmRsZS5oPgojIGluY2x1ZGUgPENvcmVGb3VuZGF0aW9uL0NGTG9jYWxlLmg+CiMgaW5jbHVkZSA8Q29yZUZvdW5kYXRpb24vQ0ZTdHJpbmcuaD4KI2VuZGlmCgojaW5jbHVkZSAibnRzdGF0dXMuaCIKI2RlZmluZSBXSU4zMl9OT19TVEFUVVMKI2luY2x1ZGUgIndpbmRlZi5oIgojaW5jbHVkZSAid2luYmFzZS5oIgojaW5jbHVkZSAid2ludXNlci5oIiAgLyogZm9yIFJUX1NUUklOR1cgKi8KI2luY2x1ZGUgIndpbnRlcm5sLmgiCiNpbmNsdWRlICJ3aW5lL3VuaWNvZGUuaCIKI2luY2x1ZGUgIndpbm5scy5oIgojaW5jbHVkZSAid2luZXJyb3IuaCIKI2luY2x1ZGUgIndpbnZlci5oIgojaW5jbHVkZSAia2VybmVsX3ByaXZhdGUuaCIKI2luY2x1ZGUgIndpbmUvZGVidWcuaCIKCldJTkVfREVGQVVMVF9ERUJVR19DSEFOTkVMKG5scyk7CgojZGVmaW5lIExPQ0FMRV9MT0NBTEVJTkZPRkxBR1NNQVNLIChMT0NBTEVfTk9VU0VST1ZFUlJJREV8TE9DQUxFX1VTRV9DUF9BQ1B8TE9DQUxFX1JFVFVSTl9OVU1CRVIpCgovKiBjdXJyZW50IGNvZGUgcGFnZXMgKi8Kc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKmFuc2lfY3B0YWJsZTsKc3RhdGljIGNvbnN0IHVuaW9uIGNwdGFibGUgKm9lbV9jcHRhYmxlOwpzdGF0aWMgY29uc3QgdW5pb24gY3B0YWJsZSAqbWFjX2NwdGFibGU7CnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICp1bml4X2NwdGFibGU7ICAvKiBOVUxMIGlmIFVURjggKi8KCnN0YXRpYyBIQU5ETEUgTkxTX1JlZ09wZW5LZXkoSEFORExFIGhSb290S2V5LCBMUENXU1RSIHN6S2V5TmFtZSk7CgpzdGF0aWMgY29uc3QgV0NIQVIgc3pObHNLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1wwJwp9OwoKc3RhdGljIGNvbnN0IFdDSEFSIHN6TG9jYWxlS2V5TmFtZVtdID0gewogICAgJ00nLCdhJywnYycsJ2gnLCdpJywnbicsJ2UnLCdcXCcsJ1MnLCd5JywncycsJ3QnLCdlJywnbScsJ1xcJywKICAgICdDJywndScsJ3InLCdyJywnZScsJ24nLCd0JywnQycsJ28nLCduJywndCcsJ3InLCdvJywnbCcsJ1MnLCdlJywndCcsJ1xcJywKICAgICdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnXFwnLCdOJywnbCcsJ3MnLCdcXCcsJ0wnLCdvJywnYycsJ2EnLCdsJywnZScsMAp9OwoKc3RhdGljIGNvbnN0IFdDSEFSIHN6Q29kZXBhZ2VLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1xcJywnQycsJ28nLCdkJywnZScsJ3AnLCdhJywnZycsJ2UnLDAKfTsKCnN0YXRpYyBjb25zdCBXQ0hBUiBzekxhbmdHcm91cHNLZXlOYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ3knLCdzJywndCcsJ2UnLCdtJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdDJywnbycsJ24nLCd0JywncicsJ28nLCdsJywnUycsJ2UnLCd0JywnXFwnLAogICAgJ0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCdcXCcsJ04nLCdsJywncycsJ1xcJywKICAgICdMJywnYScsJ24nLCdnJywndScsJ2EnLCdnJywnZScsJyAnLCdHJywncicsJ28nLCd1JywncCcsJ3MnLDAKfTsKCi8qIENoYXJzZXQgdG8gY29kZXBhZ2UgbWFwLCBzb3J0ZWQgYnkgbmFtZS4gKi8Kc3RhdGljIGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5CnsKICAgIGNvbnN0IGNoYXIgKmNoYXJzZXRfbmFtZTsKICAgIFVJTlQgICAgICAgIGNvZGVwYWdlOwp9IGNoYXJzZXRfbmFtZXNbXSA9CnsKICAgIHsgIkJJRzUiLCA5NTAgfSwKICAgIHsgIkNQMTI1MCIsIDEyNTAgfSwKICAgIHsgIkNQMTI1MSIsIDEyNTEgfSwKICAgIHsgIkNQMTI1MiIsIDEyNTIgfSwKICAgIHsgIkNQMTI1MyIsIDEyNTMgfSwKICAgIHsgIkNQMTI1NCIsIDEyNTQgfSwKICAgIHsgIkNQMTI1NSIsIDEyNTUgfSwKICAgIHsgIkNQMTI1NiIsIDEyNTYgfSwKICAgIHsgIkNQMTI1NyIsIDEyNTcgfSwKICAgIHsgIkNQMTI1OCIsIDEyNTggfSwKICAgIHsgIkNQOTMyIiwgOTMyIH0sCiAgICB7ICJDUDkzNiIsIDkzNiB9LAogICAgeyAiQ1A5NDkiLCA5NDkgfSwKICAgIHsgIkNQOTUwIiwgOTUwIH0sCiAgICB7ICJFVUNKUCIsIDIwOTMyIH0sCiAgICB7ICJHQjIzMTIiLCA5MzYgfSwKICAgIHsgIklCTTAzNyIsIDM3IH0sCiAgICB7ICJJQk0xMDI2IiwgMTAyNiB9LAogICAgeyAiSUJNNDI0IiwgNDI0IH0sCiAgICB7ICJJQk00MzciLCA0MzcgfSwKICAgIHsgIklCTTUwMCIsIDUwMCB9LAogICAgeyAiSUJNODUwIiwgODUwIH0sCiAgICB7ICJJQk04NTIiLCA4NTIgfSwKICAgIHsgIklCTTg1NSIsIDg1NSB9LAogICAgeyAiSUJNODU3IiwgODU3IH0sCiAgICB7ICJJQk04NjAiLCA4NjAgfSwKICAgIHsgIklCTTg2MSIsIDg2MSB9LAogICAgeyAiSUJNODYyIiwgODYyIH0sCiAgICB7ICJJQk04NjMiLCA4NjMgfSwKICAgIHsgIklCTTg2NCIsIDg2NCB9LAogICAgeyAiSUJNODY1IiwgODY1IH0sCiAgICB7ICJJQk04NjYiLCA4NjYgfSwKICAgIHsgIklCTTg2OSIsIDg2OSB9LAogICAgeyAiSUJNODc0IiwgODc0IH0sCiAgICB7ICJJQk04NzUiLCA4NzUgfSwKICAgIHsgIklTTzg4NTkxIiwgMjg1OTEgfSwKICAgIHsgIklTTzg4NTkxMCIsIDI4NjAwIH0sCiAgICB7ICJJU084ODU5MTMiLCAyODYwMyB9LAogICAgeyAiSVNPODg1OTE0IiwgMjg2MDQgfSwKICAgIHsgIklTTzg4NTkxNSIsIDI4NjA1IH0sCiAgICB7ICJJU084ODU5MTYiLCAyODYwNiB9LAogICAgeyAiSVNPODg1OTIiLCAyODU5MiB9LAogICAgeyAiSVNPODg1OTMiLCAyODU5MyB9LAogICAgeyAiSVNPODg1OTQiLCAyODU5NCB9LAogICAgeyAiSVNPODg1OTUiLCAyODU5NSB9LAogICAgeyAiSVNPODg1OTYiLCAyODU5NiB9LAogICAgeyAiSVNPODg1OTciLCAyODU5NyB9LAogICAgeyAiSVNPODg1OTgiLCAyODU5OCB9LAogICAgeyAiSVNPODg1OTkiLCAyODU5OSB9LAogICAgeyAiS09JOFIiLCAyMDg2NiB9LAogICAgeyAiS09JOFUiLCAyMTg2NiB9LAogICAgeyAiVVRGOCIsIENQX1VURjggfQp9OwoKCnN0cnVjdCBsb2NhbGVfbmFtZQp7CiAgICBXQ0hBUiAgd2luX25hbWVbMTI4XTsgICAvKiBXaW5kb3dzIG5hbWUgKCJlbi1VUyIpICovCiAgICBXQ0hBUiAgbGFuZ1sxMjhdOyAgICAgICAvKiBsYW5ndWFnZSAoImVuIikgKG5vdGU6IGJ1ZmZlciBjb250YWlucyB0aGUgb3RoZXIgc3RyaW5ncyB0b28pICovCiAgICBXQ0hBUiAqY291bnRyeTsgICAgICAgICAvKiBjb3VudHJ5ICgiVVMiKSAqLwogICAgV0NIQVIgKmNoYXJzZXQ7ICAgICAgICAgLyogY2hhcnNldCAoIlVURi04IikgZm9yIFVuaXggZm9ybWF0IG9ubHkgKi8KICAgIFdDSEFSICpzY3JpcHQ7ICAgICAgICAgIC8qIHNjcmlwdCAoIkxhdG4iKSBmb3IgV2luZG93cyBmb3JtYXQgb25seSAqLwogICAgV0NIQVIgKm1vZGlmaWVyOyAgICAgICAgLyogbW9kaWZpZXIgb3Igc29ydCBvcmRlciAqLwogICAgTENJRCAgIGxjaWQ7ICAgICAgICAgICAgLyogY29ycmVzcG9uZGluZyBMQ0lEICovCiAgICBpbnQgICAgbWF0Y2hlczsgICAgICAgICAvKiBudW1iZXIgb2YgZWxlbWVudHMgbWF0Y2hpbmcgTENJRCAoMC4uNCkgKi8KICAgIFVJTlQgICBjb2RlcGFnZTsgICAgICAgIC8qIGNvZGVwYWdlIGNvcnJlc3BvbmRpbmcgdG8gY2hhcnNldCAqLwp9OwoKLyogbG9jYWxlIGlkcyBjb3JyZXNwb25kaW5nIHRvIHRoZSB2YXJpb3VzIFVuaXggbG9jYWxlIHBhcmFtZXRlcnMgKi8Kc3RhdGljIExDSUQgbGNpZF9MQ19DT0xMQVRFOwpzdGF0aWMgTENJRCBsY2lkX0xDX0NUWVBFOwpzdGF0aWMgTENJRCBsY2lkX0xDX01FU1NBR0VTOwpzdGF0aWMgTENJRCBsY2lkX0xDX01PTkVUQVJZOwpzdGF0aWMgTENJRCBsY2lkX0xDX05VTUVSSUM7CnN0YXRpYyBMQ0lEIGxjaWRfTENfVElNRTsKc3RhdGljIExDSUQgbGNpZF9MQ19QQVBFUjsKc3RhdGljIExDSUQgbGNpZF9MQ19NRUFTVVJFTUVOVDsKc3RhdGljIExDSUQgbGNpZF9MQ19URUxFUEhPTkU7CgovKiBDb3B5IEFzY2lpIHN0cmluZyB0byBVbmljb2RlIHdpdGhvdXQgdXNpbmcgY29kZXBhZ2VzICovCnN0YXRpYyBpbmxpbmUgdm9pZCBzdHJjcHluQXRvVyggV0NIQVIgKmRzdCwgY29uc3QgY2hhciAqc3JjLCBzaXplX3QgbiApCnsKICAgIHdoaWxlIChuID4gMSAmJiAqc3JjKQogICAgewogICAgICAgICpkc3QrKyA9ICh1bnNpZ25lZCBjaGFyKSpzcmMrKzsKICAgICAgICBuLS07CiAgICB9CiAgICBpZiAobikgKmRzdCA9IDA7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWdldF9sY2lkX2NvZGVwYWdlCiAqCiAqIFJldHJpZXZlIHRoZSBBTlNJIGNvZGVwYWdlIGZvciBhIGdpdmVuIGxvY2FsZS4KICovCnN0YXRpYyBpbmxpbmUgVUlOVCBnZXRfbGNpZF9jb2RlcGFnZSggTENJRCBsY2lkICkKewogICAgVUlOVCByZXQ7CiAgICBpZiAoIUdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfSURFRkFVTFRBTlNJQ09ERVBBR0V8TE9DQUxFX1JFVFVSTl9OVU1CRVIsIChXQ0hBUiAqKSZyZXQsCiAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YocmV0KS9zaXplb2YoV0NIQVIpICkpIHJldCA9IDA7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfY29kZXBhZ2VfdGFibGUKICoKICogRmluZCB0aGUgdGFibGUgZm9yIGEgZ2l2ZW4gY29kZXBhZ2UsIGhhbmRsaW5nIENQX0FDUCBldGMuIHBzZXVkby1jb2RlcGFnZXMKICovCnN0YXRpYyBjb25zdCB1bmlvbiBjcHRhYmxlICpnZXRfY29kZXBhZ2VfdGFibGUoIHVuc2lnbmVkIGludCBjb2RlcGFnZSApCnsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnJldCA9IE5VTEw7CgogICAgYXNzZXJ0KCBhbnNpX2NwdGFibGUgKTsgIC8qIGluaXQgbXVzdCBoYXZlIGJlZW4gZG9uZSBhbHJlYWR5ICovCgogICAgc3dpdGNoKGNvZGVwYWdlKQogICAgewogICAgY2FzZSBDUF9BQ1A6CiAgICAgICAgcmV0dXJuIGFuc2lfY3B0YWJsZTsKICAgIGNhc2UgQ1BfT0VNQ1A6CiAgICAgICAgcmV0dXJuIG9lbV9jcHRhYmxlOwogICAgY2FzZSBDUF9NQUNDUDoKICAgICAgICByZXR1cm4gbWFjX2NwdGFibGU7CiAgICBjYXNlIENQX1VURjc6CiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENQX1RIUkVBRF9BQ1A6CiAgICAgICAgaWYgKCEoY29kZXBhZ2UgPSBrZXJuZWxfZ2V0X3RocmVhZF9kYXRhKCktPmNvZGVfcGFnZSkpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBkZWZhdWx0OgogICAgICAgIGlmIChjb2RlcGFnZSA9PSBhbnNpX2NwdGFibGUtPmluZm8uY29kZXBhZ2UpIHJldHVybiBhbnNpX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG9lbV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gb2VtX2NwdGFibGU7CiAgICAgICAgaWYgKGNvZGVwYWdlID09IG1hY19jcHRhYmxlLT5pbmZvLmNvZGVwYWdlKSByZXR1cm4gbWFjX2NwdGFibGU7CiAgICAgICAgcmV0ID0gd2luZV9jcF9nZXRfdGFibGUoIGNvZGVwYWdlICk7CiAgICAgICAgYnJlYWs7CiAgICB9CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBjaGFyc2V0X2NtcCAoaW50ZXJuYWwpCiAqLwpzdGF0aWMgaW50IGNoYXJzZXRfY21wKCBjb25zdCB2b2lkICpuYW1lLCBjb25zdCB2b2lkICplbnRyeSApCnsKICAgIGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICpjaGFyc2V0ID0gKGNvbnN0IHN0cnVjdCBjaGFyc2V0X2VudHJ5ICopZW50cnk7CiAgICByZXR1cm4gc3RyY2FzZWNtcCggKGNvbnN0IGNoYXIgKiluYW1lLCBjaGFyc2V0LT5jaGFyc2V0X25hbWUgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlmaW5kX2NoYXJzZXQKICovCnN0YXRpYyBVSU5UIGZpbmRfY2hhcnNldCggY29uc3QgV0NIQVIgKm5hbWUgKQp7CiAgICBjb25zdCBzdHJ1Y3QgY2hhcnNldF9lbnRyeSAqZW50cnk7CiAgICBjaGFyIGNoYXJzZXRfbmFtZVsxNl07CiAgICBzaXplX3QgaSwgajsKCiAgICAvKiByZW1vdmUgcHVuY3R1YXRpb24gY2hhcmFjdGVycyBmcm9tIGNoYXJzZXQgbmFtZSAqLwogICAgZm9yIChpID0gaiA9IDA7IG5hbWVbaV0gJiYgaiA8IHNpemVvZihjaGFyc2V0X25hbWUpLTE7IGkrKykKICAgICAgICBpZiAoaXNhbG51bSgodW5zaWduZWQgY2hhciluYW1lW2ldKSkgY2hhcnNldF9uYW1lW2orK10gPSBuYW1lW2ldOwogICAgY2hhcnNldF9uYW1lW2pdID0gMDsKCiAgICBlbnRyeSA9IGJzZWFyY2goIGNoYXJzZXRfbmFtZSwgY2hhcnNldF9uYW1lcywKICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXMpL3NpemVvZihjaGFyc2V0X25hbWVzWzBdKSwKICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGNoYXJzZXRfbmFtZXNbMF0pLCBjaGFyc2V0X2NtcCApOwogICAgaWYgKGVudHJ5KSByZXR1cm4gZW50cnktPmNvZGVwYWdlOwogICAgcmV0dXJuIDA7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGZpbmRfbG9jYWxlX2lkX2NhbGxiYWNrCiAqLwpzdGF0aWMgQk9PTCBDQUxMQkFDSyBmaW5kX2xvY2FsZV9pZF9jYWxsYmFjayggSE1PRFVMRSBoTW9kdWxlLCBMUENXU1RSIHR5cGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENXU1RSIG5hbWUsIFdPUkQgTGFuZ0lELCBMUEFSQU0gbFBhcmFtICkKewogICAgc3RydWN0IGxvY2FsZV9uYW1lICpkYXRhID0gKHN0cnVjdCBsb2NhbGVfbmFtZSAqKWxQYXJhbTsKICAgIFdDSEFSIGJ1ZmZlclsxMjhdOwogICAgaW50IG1hdGNoZXMgPSAwOwogICAgTENJRCBsY2lkID0gTUFLRUxDSUQoIExhbmdJRCwgU09SVF9ERUZBVUxUICk7ICAvKiBGSVhNRTogaGFuZGxlIHNvcnQgb3JkZXIgKi8KCiAgICBpZiAoUFJJTUFSWUxBTkdJRChMYW5nSUQpID09IExBTkdfTkVVVFJBTCkgcmV0dXJuIFRSVUU7IC8qIGNvbnRpbnVlIHNlYXJjaCAqLwoKICAgIC8qIGZpcnN0IGNoZWNrIGV4YWN0IG5hbWUgKi8KICAgIGlmIChkYXRhLT53aW5fbmFtZVswXSAmJgogICAgICAgIEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU05BTUUgfCBMT0NBTEVfTk9VU0VST1ZFUlJJREUsCiAgICAgICAgICAgICAgICAgICAgICAgIGJ1ZmZlciwgc2l6ZW9mKGJ1ZmZlcikvc2l6ZW9mKFdDSEFSKSApKQogICAgewogICAgICAgIGlmICghc3RyY21wVyggZGF0YS0+d2luX25hbWUsIGJ1ZmZlciApKQogICAgICAgIHsKICAgICAgICAgICAgbWF0Y2hlcyA9IDQ7ICAvKiBldmVyeXRoaW5nIG1hdGNoZXMgKi8KICAgICAgICAgICAgZ290byBkb25lOwogICAgICAgIH0KICAgIH0KCiAgICBpZiAoIUdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU0lTTzYzOUxBTkdOQU1FIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLAogICAgICAgICAgICAgICAgICAgICAgICAgYnVmZmVyLCBzaXplb2YoYnVmZmVyKS9zaXplb2YoV0NIQVIpICkpCiAgICAgICAgcmV0dXJuIFRSVUU7CiAgICBpZiAoc3RyY21wVyggYnVmZmVyLCBkYXRhLT5sYW5nICkpIHJldHVybiBUUlVFOwogICAgbWF0Y2hlcysrOyAgLyogbGFuZ3VhZ2UgbmFtZSBtYXRjaGVkICovCgogICAgaWYgKGRhdGEtPmNvdW50cnkpCiAgICB7CiAgICAgICAgaWYgKEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU0lTTzMxNjZDVFJZTkFNRXxMT0NBTEVfTk9VU0VST1ZFUlJJREUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBidWZmZXIsIHNpemVvZihidWZmZXIpL3NpemVvZihXQ0hBUikgKSkKICAgICAgICB7CiAgICAgICAgICAgIGlmIChzdHJjbXBXKCBidWZmZXIsIGRhdGEtPmNvdW50cnkgKSkgZ290byBkb25lOwogICAgICAgICAgICBtYXRjaGVzKys7ICAvKiBjb3VudHJ5IG5hbWUgbWF0Y2hlZCAqLwogICAgICAgIH0KICAgIH0KICAgIGVsc2UgIC8qIG1hdGNoIGRlZmF1bHQgbGFuZ3VhZ2UgKi8KICAgIHsKICAgICAgICBpZiAoU1VCTEFOR0lEKExhbmdJRCkgPT0gU1VCTEFOR19ERUZBVUxUKSBtYXRjaGVzKys7CiAgICB9CgogICAgaWYgKGRhdGEtPmNvZGVwYWdlKQogICAgewogICAgICAgIFVJTlQgdW5peF9jcDsKICAgICAgICBpZiAoR2V0TG9jYWxlSW5mb1coIGxjaWQsIExPQ0FMRV9JREVGQVVMVFVOSVhDT0RFUEFHRSB8IExPQ0FMRV9SRVRVUk5fTlVNQkVSLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQV1NUUikmdW5peF9jcCwgc2l6ZW9mKHVuaXhfY3ApL3NpemVvZihXQ0hBUikgKSkKICAgICAgICB7CiAgICAgICAgICAgIGlmICh1bml4X2NwID09IGRhdGEtPmNvZGVwYWdlKSBtYXRjaGVzKys7CiAgICAgICAgfQogICAgfQoKICAgIC8qIEZJWE1FOiBjaGVjayBzb3J0IG9yZGVyICovCgpkb25lOgogICAgaWYgKG1hdGNoZXMgPiBkYXRhLT5tYXRjaGVzKQogICAgewogICAgICAgIGRhdGEtPmxjaWQgPSBsY2lkOwogICAgICAgIGRhdGEtPm1hdGNoZXMgPSBtYXRjaGVzOwogICAgfQogICAgcmV0dXJuIChkYXRhLT5tYXRjaGVzIDwgNCk7ICAvKiBubyBuZWVkIHRvIGNvbnRpbnVlIGZvciBwZXJmZWN0IG1hdGNoICovCn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCXBhcnNlX2xvY2FsZV9uYW1lCiAqCiAqIFBhcnNlIGEgbG9jYWxlIG5hbWUgaW50byBhIHN0cnVjdCBsb2NhbGVfbmFtZSwgaGFuZGxpbmcgYm90aCBXaW5kb3dzIGFuZCBVbml4IGZvcm1hdHMuCiAqIFVuaXggZm9ybWF0IGlzOiBsYW5nW19jb3VudHJ5XVsuY2hhcnNldF1bQG1vZGlmaWVyXQogKiBXaW5kb3dzIGZvcm1hdCBpczogbGFuZ1stc2NyaXB0XVstY291bnRyeV1bX21vZGlmaWVyXQogKi8Kc3RhdGljIHZvaWQgcGFyc2VfbG9jYWxlX25hbWUoIGNvbnN0IFdDSEFSICpzdHIsIHN0cnVjdCBsb2NhbGVfbmFtZSAqbmFtZSApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzZXBXW10gPSB7Jy0nLCdfJywnLicsJ0AnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHdpbnNlcFdbXSA9IHsnLScsJ18nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHBvc2l4V1tdID0geydQJywnTycsJ1MnLCdJJywnWCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgY1dbXSA9IHsnQycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGF0aW5XW10gPSB7J2wnLCdhJywndCcsJ2knLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsYXRuV1tdID0geyctJywnTCcsJ2EnLCd0JywnbicsMH07CiAgICBXQ0hBUiAqcDsKCiAgICBuYW1lLT5jb3VudHJ5ID0gbmFtZS0+Y2hhcnNldCA9IG5hbWUtPnNjcmlwdCA9IG5hbWUtPm1vZGlmaWVyID0gTlVMTDsKICAgIG5hbWUtPmxjaWQgPSBNQUtFTENJRCggTUFLRUxBTkdJRChMQU5HX0VOR0xJU0gsU1VCTEFOR19ERUZBVUxUKSwgU09SVF9ERUZBVUxUICk7CiAgICBuYW1lLT5tYXRjaGVzID0gMDsKICAgIG5hbWUtPmNvZGVwYWdlID0gMDsKICAgIG5hbWUtPndpbl9uYW1lWzBdID0gMDsKICAgIGxzdHJjcHluVyggbmFtZS0+bGFuZywgc3RyLCBzaXplb2YobmFtZS0+bGFuZykvc2l6ZW9mKFdDSEFSKSApOwoKICAgIGlmICghKHAgPSBzdHJwYnJrVyggbmFtZS0+bGFuZywgc2VwVyApKSkKICAgIHsKICAgICAgICBpZiAoIXN0cmNtcFcoIG5hbWUtPmxhbmcsIHBvc2l4VyApIHx8ICFzdHJjbXBXKCBuYW1lLT5sYW5nLCBjVyApKQogICAgICAgIHsKICAgICAgICAgICAgbmFtZS0+bWF0Y2hlcyA9IDQ7ICAvKiBwZXJmZWN0IG1hdGNoIGZvciBkZWZhdWx0IEVuZ2xpc2ggbGNpZCAqLwogICAgICAgICAgICByZXR1cm47CiAgICAgICAgfQogICAgICAgIHN0cmNweVcoIG5hbWUtPndpbl9uYW1lLCBuYW1lLT5sYW5nICk7CiAgICB9CiAgICBlbHNlIGlmICgqcCA9PSAnLScpICAvKiBXaW5kb3dzIGZvcm1hdCAqLwogICAgewogICAgICAgIHN0cmNweVcoIG5hbWUtPndpbl9uYW1lLCBuYW1lLT5sYW5nICk7CiAgICAgICAgKnArKyA9IDA7CiAgICAgICAgbmFtZS0+Y291bnRyeSA9IHA7CiAgICAgICAgaWYgKCEocCA9IHN0cnBicmtXKCBwLCB3aW5zZXBXICkpKSBnb3RvIGRvbmU7CiAgICAgICAgaWYgKCpwID09ICctJykKICAgICAgICB7CiAgICAgICAgICAgICpwKysgPSAwOwogICAgICAgICAgICBuYW1lLT5zY3JpcHQgPSBuYW1lLT5jb3VudHJ5OwogICAgICAgICAgICBuYW1lLT5jb3VudHJ5ID0gcDsKICAgICAgICAgICAgaWYgKCEocCA9IHN0cnBicmtXKCBwLCB3aW5zZXBXICkpKSBnb3RvIGRvbmU7CiAgICAgICAgfQogICAgICAgICpwKysgPSAwOwogICAgICAgIG5hbWUtPm1vZGlmaWVyID0gcDsKICAgIH0KICAgIGVsc2UgIC8qIFVuaXggZm9ybWF0ICovCiAgICB7CiAgICAgICAgaWYgKCpwID09ICdfJykKICAgICAgICB7CiAgICAgICAgICAgICpwKysgPSAwOwogICAgICAgICAgICBuYW1lLT5jb3VudHJ5ID0gcDsKICAgICAgICAgICAgcCA9IHN0cnBicmtXKCBwLCBzZXBXICsgMiApOwogICAgICAgIH0KICAgICAgICBpZiAocCAmJiAqcCA9PSAnLicpCiAgICAgICAgewogICAgICAgICAgICAqcCsrID0gMDsKICAgICAgICAgICAgbmFtZS0+Y2hhcnNldCA9IHA7CiAgICAgICAgICAgIG5hbWUtPmNvZGVwYWdlID0gZmluZF9jaGFyc2V0KCBuYW1lLT5jaGFyc2V0ICk7CiAgICAgICAgICAgIHAgPSBzdHJjaHJXKCBwLCAnQCcgKTsKICAgICAgICB9CiAgICAgICAgaWYgKHApCiAgICAgICAgewogICAgICAgICAgICAqcCsrID0gMDsKICAgICAgICAgICAgbmFtZS0+bW9kaWZpZXIgPSBwOwogICAgICAgIH0KCiAgICAgICAgLyogcmVidWlsZCBhIFdpbmRvd3MgbmFtZSBpZiBwb3NzaWJsZSAqLwoKICAgICAgICBpZiAobmFtZS0+Y2hhcnNldCkgZ290byBkb25lOyAgLyogY2FuJ3Qgc3BlY2lmeSBjaGFyc2V0IGluIFdpbmRvd3MgZm9ybWF0ICovCiAgICAgICAgaWYgKG5hbWUtPm1vZGlmaWVyICYmIHN0cmNtcFcoIG5hbWUtPm1vZGlmaWVyLCBsYXRpblcgKSkKICAgICAgICAgICAgZ290byBkb25lOyAgLyogb25seSBMYXRuIHNjcmlwdCBzdXBwb3J0ZWQgZm9yIG5vdyAqLwogICAgICAgIHN0cmNweVcoIG5hbWUtPndpbl9uYW1lLCBuYW1lLT5sYW5nICk7CiAgICAgICAgaWYgKG5hbWUtPm1vZGlmaWVyKSBzdHJjYXRXKCBuYW1lLT53aW5fbmFtZSwgbGF0blcgKTsKICAgICAgICBpZiAobmFtZS0+Y291bnRyeSkKICAgICAgICB7CiAgICAgICAgICAgIHAgPSBuYW1lLT53aW5fbmFtZSArIHN0cmxlblcobmFtZS0+d2luX25hbWUpOwogICAgICAgICAgICAqcCsrID0gJy0nOwogICAgICAgICAgICBzdHJjcHlXKCBwLCBuYW1lLT5jb3VudHJ5ICk7CiAgICAgICAgfQogICAgfQpkb25lOgogICAgRW51bVJlc291cmNlTGFuZ3VhZ2VzVygga2VybmVsMzJfaGFuZGxlLCAoTFBDV1NUUilSVF9TVFJJTkcsIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBmaW5kX2xvY2FsZV9pZF9jYWxsYmFjaywgKExQQVJBTSluYW1lICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGNvbnZlcnRfZGVmYXVsdF9sY2lkCiAqCiAqIEdldCB0aGUgZGVmYXVsdCBMQ0lEIHRvIHVzZSBmb3IgYSBnaXZlbiBsY3R5cGUgaW4gR2V0TG9jYWxlSW5mby4KICovCnN0YXRpYyBMQ0lEIGNvbnZlcnRfZGVmYXVsdF9sY2lkKCBMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUgKQp7CiAgICBpZiAobGNpZCA9PSBMT0NBTEVfU1lTVEVNX0RFRkFVTFQgfHwKICAgICAgICBsY2lkID09IExPQ0FMRV9VU0VSX0RFRkFVTFQgfHwKICAgICAgICBsY2lkID09IExPQ0FMRV9ORVVUUkFMKQogICAgewogICAgICAgIExDSUQgZGVmYXVsdF9pZCA9IDA7CgogICAgICAgIHN3aXRjaChsY3R5cGUgJiAweGZmZmYpCiAgICAgICAgewogICAgICAgIGNhc2UgTE9DQUxFX1NTT1JUTkFNRToKICAgICAgICAgICAgZGVmYXVsdF9pZCA9IGxjaWRfTENfQ09MTEFURTsKICAgICAgICAgICAgYnJlYWs7CgogICAgICAgIGNhc2UgTE9DQUxFX0ZPTlRTSUdOQVRVUkU6CiAgICAgICAgY2FzZSBMT0NBTEVfSURFRkFVTFRBTlNJQ09ERVBBR0U6CiAgICAgICAgY2FzZSBMT0NBTEVfSURFRkFVTFRDT0RFUEFHRToKICAgICAgICBjYXNlIExPQ0FMRV9JREVGQVVMVEVCQ0RJQ0NPREVQQUdFOgogICAgICAgIGNhc2UgTE9DQUxFX0lERUZBVUxUTUFDQ09ERVBBR0U6CiAgICAgICAgY2FzZSBMT0NBTEVfSURFRkFVTFRVTklYQ09ERVBBR0U6CiAgICAgICAgICAgIGRlZmF1bHRfaWQgPSBsY2lkX0xDX0NUWVBFOwogICAgICAgICAgICBicmVhazsKCiAgICAgICAgY2FzZSBMT0NBTEVfSUNVUlJESUdJVFM6CiAgICAgICAgY2FzZSBMT0NBTEVfSUNVUlJFTkNZOgogICAgICAgIGNhc2UgTE9DQUxFX0lJTlRMQ1VSUkRJR0lUUzoKICAgICAgICBjYXNlIExPQ0FMRV9JTkVHQ1VSUjoKICAgICAgICBjYXNlIExPQ0FMRV9JTkVHU0VQQllTUEFDRToKICAgICAgICBjYXNlIExPQ0FMRV9JTkVHU0lHTlBPU046CiAgICAgICAgY2FzZSBMT0NBTEVfSU5FR1NZTVBSRUNFREVTOgogICAgICAgIGNhc2UgTE9DQUxFX0lQT1NTRVBCWVNQQUNFOgogICAgICAgIGNhc2UgTE9DQUxFX0lQT1NTSUdOUE9TTjoKICAgICAgICBjYXNlIExPQ0FMRV9JUE9TU1lNUFJFQ0VERVM6CiAgICAgICAgY2FzZSBMT0NBTEVfU0NVUlJFTkNZOgogICAgICAgIGNhc2UgTE9DQUxFX1NJTlRMU1lNQk9MOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05ERUNJTUFMU0VQOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05HUk9VUElORzoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhPVVNBTkRTRVA6CiAgICAgICAgY2FzZSBMT0NBTEVfU05BVElWRUNVUlJOQU1FOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19NT05FVEFSWTsKICAgICAgICAgICAgYnJlYWs7CgogICAgICAgIGNhc2UgTE9DQUxFX0lESUdJVFM6CiAgICAgICAgY2FzZSBMT0NBTEVfSURJR0lUU1VCU1RJVFVUSU9OOgogICAgICAgIGNhc2UgTE9DQUxFX0lMWkVSTzoKICAgICAgICBjYXNlIExPQ0FMRV9JTkVHTlVNQkVSOgogICAgICAgIGNhc2UgTE9DQUxFX1NERUNJTUFMOgogICAgICAgIGNhc2UgTE9DQUxFX1NHUk9VUElORzoKICAgICAgICBjYXNlIExPQ0FMRV9TTkFOOgogICAgICAgIGNhc2UgTE9DQUxFX1NOQVRJVkVESUdJVFM6CiAgICAgICAgY2FzZSBMT0NBTEVfU05FR0FUSVZFU0lHTjoKICAgICAgICBjYXNlIExPQ0FMRV9TTkVHSU5GSU5JVFk6CiAgICAgICAgY2FzZSBMT0NBTEVfU1BPU0lORklOSVRZOgogICAgICAgIGNhc2UgTE9DQUxFX1NQT1NJVElWRVNJR046CiAgICAgICAgY2FzZSBMT0NBTEVfU1RIT1VTQU5EOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19OVU1FUklDOwogICAgICAgICAgICBicmVhazsKCiAgICAgICAgY2FzZSBMT0NBTEVfSUNBTEVOREFSVFlQRToKICAgICAgICBjYXNlIExPQ0FMRV9JQ0VOVFVSWToKICAgICAgICBjYXNlIExPQ0FMRV9JREFURToKICAgICAgICBjYXNlIExPQ0FMRV9JREFZTFpFUk86CiAgICAgICAgY2FzZSBMT0NBTEVfSUZJUlNUREFZT0ZXRUVLOgogICAgICAgIGNhc2UgTE9DQUxFX0lGSVJTVFdFRUtPRllFQVI6CiAgICAgICAgY2FzZSBMT0NBTEVfSUxEQVRFOgogICAgICAgIGNhc2UgTE9DQUxFX0lNT05MWkVSTzoKICAgICAgICBjYXNlIExPQ0FMRV9JT1BUSU9OQUxDQUxFTkRBUjoKICAgICAgICBjYXNlIExPQ0FMRV9JVElNRToKICAgICAgICBjYXNlIExPQ0FMRV9JVElNRU1BUktQT1NOOgogICAgICAgIGNhc2UgTE9DQUxFX0lUTFpFUk86CiAgICAgICAgY2FzZSBMT0NBTEVfUzExNTk6CiAgICAgICAgY2FzZSBMT0NBTEVfUzIzNTk6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVkRBWU5BTUUxOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZEQVlOQU1FMjoKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWREFZTkFNRTM6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVkRBWU5BTUU0OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZEQVlOQU1FNToKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWREFZTkFNRTY6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVkRBWU5BTUU3OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUxOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUyOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUzOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU0OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU1OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU2OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU3OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU4OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUU5OgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUxMDoKICAgICAgICBjYXNlIExPQ0FMRV9TQUJCUkVWTU9OVEhOQU1FMTE6CiAgICAgICAgY2FzZSBMT0NBTEVfU0FCQlJFVk1PTlRITkFNRTEyOgogICAgICAgIGNhc2UgTE9DQUxFX1NBQkJSRVZNT05USE5BTUUxMzoKICAgICAgICBjYXNlIExPQ0FMRV9TREFURToKICAgICAgICBjYXNlIExPQ0FMRV9TREFZTkFNRTE6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RBWU5BTUUyOgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVlOQU1FMzoKICAgICAgICBjYXNlIExPQ0FMRV9TREFZTkFNRTQ6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RBWU5BTUU1OgogICAgICAgIGNhc2UgTE9DQUxFX1NEQVlOQU1FNjoKICAgICAgICBjYXNlIExPQ0FMRV9TREFZTkFNRTc6CiAgICAgICAgY2FzZSBMT0NBTEVfU0RVUkFUSU9OOgogICAgICAgIGNhc2UgTE9DQUxFX1NMT05HREFURToKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMToKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMjoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMzoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FNDoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FNToKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FNjoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FNzoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FODoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FOToKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMTA6CiAgICAgICAgY2FzZSBMT0NBTEVfU01PTlRITkFNRTExOgogICAgICAgIGNhc2UgTE9DQUxFX1NNT05USE5BTUUxMjoKICAgICAgICBjYXNlIExPQ0FMRV9TTU9OVEhOQU1FMTM6CiAgICAgICAgY2FzZSBMT0NBTEVfU1NIT1JUREFURToKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FMToKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FMjoKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FMzoKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FNDoKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FNToKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FNjoKICAgICAgICBjYXNlIExPQ0FMRV9TU0hPUlRFU1REQVlOQU1FNzoKICAgICAgICBjYXNlIExPQ0FMRV9TVElNRToKICAgICAgICBjYXNlIExPQ0FMRV9TVElNRUZPUk1BVDoKICAgICAgICBjYXNlIExPQ0FMRV9TWUVBUk1PTlRIOgogICAgICAgICAgICBkZWZhdWx0X2lkID0gbGNpZF9MQ19USU1FOwogICAgICAgICAgICBicmVhazsKCiAgICAgICAgY2FzZSBMT0NBTEVfSVBBUEVSU0laRToKICAgICAgICAgICAgZGVmYXVsdF9pZCA9IGxjaWRfTENfUEFQRVI7CiAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICBjYXNlIExPQ0FMRV9JTUVBU1VSRToKICAgICAgICAgICAgZGVmYXVsdF9pZCA9IGxjaWRfTENfTUVBU1VSRU1FTlQ7CiAgICAgICAgICAgIGJyZWFrOwoKICAgICAgICBjYXNlIExPQ0FMRV9JQ09VTlRSWToKICAgICAgICAgICAgZGVmYXVsdF9pZCA9IGxjaWRfTENfVEVMRVBIT05FOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgaWYgKGRlZmF1bHRfaWQpIGxjaWQgPSBkZWZhdWx0X2lkOwogICAgfQogICAgcmV0dXJuIENvbnZlcnREZWZhdWx0TG9jYWxlKCBsY2lkICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCWNyZWF0ZV9yZWdpc3RyeV9rZXkKICoKICogQ3JlYXRlIHRoZSBDb250cm9sIFBhbmVsXFxJbnRlcm5hdGlvbmFsIHJlZ2lzdHJ5IGtleS4KICovCnN0YXRpYyBpbmxpbmUgSEFORExFIGNyZWF0ZV9yZWdpc3RyeV9rZXkodm9pZCkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGludGxXW10gPSB7J0MnLCdvJywnbicsJ3QnLCdyJywnbycsJ2wnLCcgJywnUCcsJ2EnLCduJywnZScsJ2wnLCdcXCcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAnSScsJ24nLCd0JywnZScsJ3InLCduJywnYScsJ3QnLCdpJywnbycsJ24nLCdhJywnbCcsMH07CiAgICBPQkpFQ1RfQVRUUklCVVRFUyBhdHRyOwogICAgVU5JQ09ERV9TVFJJTkcgbmFtZVc7CiAgICBIQU5ETEUgaGtleTsKCiAgICBpZiAoUnRsT3BlbkN1cnJlbnRVc2VyKCBLRVlfQUxMX0FDQ0VTUywgJmhrZXkgKSAhPSBTVEFUVVNfU1VDQ0VTUykgcmV0dXJuIDA7CgogICAgYXR0ci5MZW5ndGggPSBzaXplb2YoYXR0cik7CiAgICBhdHRyLlJvb3REaXJlY3RvcnkgPSBoa2V5OwogICAgYXR0ci5PYmplY3ROYW1lID0gJm5hbWVXOwogICAgYXR0ci5BdHRyaWJ1dGVzID0gMDsKICAgIGF0dHIuU2VjdXJpdHlEZXNjcmlwdG9yID0gTlVMTDsKICAgIGF0dHIuU2VjdXJpdHlRdWFsaXR5T2ZTZXJ2aWNlID0gTlVMTDsKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmbmFtZVcsIGludGxXICk7CgogICAgaWYgKE50Q3JlYXRlS2V5KCAmaGtleSwgS0VZX0FMTF9BQ0NFU1MsICZhdHRyLCAwLCBOVUxMLCAwLCBOVUxMICkgIT0gU1RBVFVTX1NVQ0NFU1MpIGhrZXkgPSAwOwogICAgTnRDbG9zZSggYXR0ci5Sb290RGlyZWN0b3J5ICk7CiAgICByZXR1cm4gaGtleTsKfQoKCi8qIHVwZGF0ZSB0aGUgcmVnaXN0cnkgc2V0dGluZ3MgZm9yIGEgZ2l2ZW4gbG9jYWxlIHBhcmFtZXRlciAqLwovKiByZXR1cm4gVFJVRSBpZiBhbiB1cGRhdGUgd2FzIG5lZWRlZCAqLwpzdGF0aWMgQk9PTCBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBIS0VZIGhrZXksIGNvbnN0IFdDSEFSICpuYW1lLCBMQ0lEIGxjaWQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IExDVFlQRSAqdmFsdWVzLCBVSU5UIG5iX3ZhbHVlcyApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBmb3JtYXRXW10gPSB7ICclJywnMCcsJzgnLCd4JywwIH07CiAgICBXQ0hBUiBidWZmZXJXWzQwXTsKICAgIFVOSUNPREVfU1RSSU5HIG5hbWVXOwogICAgRFdPUkQgY291bnQsIGk7CgogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgbmFtZSApOwogICAgY291bnQgPSBzaXplb2YoYnVmZmVyVyk7CiAgICBpZiAoIU50UXVlcnlWYWx1ZUtleShoa2V5LCAmbmFtZVcsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLCAoTFBCWVRFKWJ1ZmZlclcsIGNvdW50LCAmY291bnQpKQogICAgewogICAgICAgIGNvbnN0IEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICppbmZvID0gKEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICopYnVmZmVyVzsKICAgICAgICBMUENXU1RSIHRleHQgPSAoTFBDV1NUUilpbmZvLT5EYXRhOwoKICAgICAgICBpZiAoc3RydG91bFcoIHRleHQsIE5VTEwsIDE2ICkgPT0gbGNpZCkgcmV0dXJuIEZBTFNFOyAvKiBhbHJlYWR5IHNldCBjb3JyZWN0bHkgKi8KICAgICAgICBUUkFDRSggInVwZGF0aW5nIHJlZ2lzdHJ5LCBsb2NhbGUgJXMgY2hhbmdlZCAlcyAtPiAlMDh4XG4iLAogICAgICAgICAgICAgICBkZWJ1Z3N0cl93KG5hbWUpLCBkZWJ1Z3N0cl93KHRleHQpLCBsY2lkICk7CiAgICB9CiAgICBlbHNlIFRSQUNFKCAidXBkYXRpbmcgcmVnaXN0cnksIGxvY2FsZSAlcyBjaGFuZ2VkIG5vbmUgLT4gJTA4eFxuIiwgZGVidWdzdHJfdyhuYW1lKSwgbGNpZCApOwogICAgc3ByaW50ZlcoIGJ1ZmZlclcsIGZvcm1hdFcsIGxjaWQgKTsKICAgIE50U2V0VmFsdWVLZXkoIGhrZXksICZuYW1lVywgMCwgUkVHX1NaLCBidWZmZXJXLCAoc3RybGVuVyhidWZmZXJXKSArIDEpICogc2l6ZW9mKFdDSEFSKSApOwoKICAgIGZvciAoaSA9IDA7IGkgPCBuYl92YWx1ZXM7IGkrKykKICAgIHsKICAgICAgICBHZXRMb2NhbGVJbmZvVyggbGNpZCwgdmFsdWVzW2ldIHwgTE9DQUxFX05PVVNFUk9WRVJSSURFLCBidWZmZXJXLAogICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoYnVmZmVyVykvc2l6ZW9mKFdDSEFSKSApOwogICAgICAgIFNldExvY2FsZUluZm9XKCBsY2lkLCB2YWx1ZXNbaV0sIGJ1ZmZlclcgKTsKICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlMT0NBTEVfSW5pdFJlZ2lzdHJ5CiAqCiAqIFVwZGF0ZSByZWdpc3RyeSBjb250ZW50cyBvbiBzdGFydHVwIGlmIHRoZSB1c2VyIGxvY2FsZSBoYXMgY2hhbmdlZC4KICogVGhpcyBzaW11bGF0ZXMgdGhlIGFjdGlvbiBvZiB0aGUgV2luZG93cyBjb250cm9sIHBhbmVsLgogKi8Kdm9pZCBMT0NBTEVfSW5pdFJlZ2lzdHJ5KHZvaWQpCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBhY3BXW10gPSB7J0EnLCdDJywnUCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgb2VtY3BXW10gPSB7J08nLCdFJywnTScsJ0MnLCdQJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBtYWNjcFdbXSA9IHsnTScsJ0EnLCdDJywnQycsJ1AnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGxvY2FsZVdbXSA9IHsnTCcsJ28nLCdjJywnYScsJ2wnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY19jdHlwZVdbXSA9IHsgJ0wnLCdDJywnXycsJ0MnLCdUJywnWScsJ1AnLCdFJywwIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGNfbW9uZXRhcnlXW10gPSB7ICdMJywnQycsJ18nLCdNJywnTycsJ04nLCdFJywnVCcsJ0EnLCdSJywnWScsMCB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGxjX251bWVyaWNXW10gPSB7ICdMJywnQycsJ18nLCdOJywnVScsJ00nLCdFJywnUicsJ0knLCdDJywwIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGNfdGltZVdbXSA9IHsgJ0wnLCdDJywnXycsJ1QnLCdJJywnTScsJ0UnLDAgfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBsY19tZWFzdXJlbWVudFdbXSA9IHsgJ0wnLCdDJywnXycsJ00nLCdFJywnQScsJ1MnLCdVJywnUicsJ0UnLCdNJywnRScsJ04nLCdUJywwIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGNfdGVsZXBob25lV1tdID0geyAnTCcsJ0MnLCdfJywnVCcsJ0UnLCdMJywnRScsJ1AnLCdIJywnTycsJ04nLCdFJywwIH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgbGNfcGFwZXJXW10gPSB7ICdMJywnQycsJ18nLCdQJywnQScsJ1AnLCdFJywnUicsMH07CiAgICBzdGF0aWMgY29uc3Qgc3RydWN0CiAgICB7CiAgICAgICAgTFBDV1NUUiBuYW1lOwogICAgICAgIFVTSE9SVCB2YWx1ZTsKICAgIH0gdXBkYXRlX2NwX3ZhbHVlc1tdID0gewogICAgICAgIHsgYWNwVywgTE9DQUxFX0lERUZBVUxUQU5TSUNPREVQQUdFIH0sCiAgICAgICAgeyBvZW1jcFcsIExPQ0FMRV9JREVGQVVMVENPREVQQUdFIH0sCiAgICAgICAgeyBtYWNjcFcsIExPQ0FMRV9JREVGQVVMVE1BQ0NPREVQQUdFIH0KICAgIH07CiAgICBzdGF0aWMgY29uc3QgTENUWVBFIGxjX21lc3NhZ2VzX3ZhbHVlc1tdID0gewogICAgICBMT0NBTEVfU0xBTkdVQUdFLAogICAgICBMT0NBTEVfU0NPVU5UUlksCiAgICAgIExPQ0FMRV9TTElTVCB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY19tb25ldGFyeV92YWx1ZXNbXSA9IHsKICAgICAgTE9DQUxFX1NDVVJSRU5DWSwKICAgICAgTE9DQUxFX0lDVVJSRU5DWSwKICAgICAgTE9DQUxFX0lORUdDVVJSLAogICAgICBMT0NBTEVfSUNVUlJESUdJVFMsCiAgICAgIExPQ0FMRV9JTFpFUk8sCiAgICAgIExPQ0FMRV9TTU9OREVDSU1BTFNFUCwKICAgICAgTE9DQUxFX1NNT05HUk9VUElORywKICAgICAgTE9DQUxFX1NNT05USE9VU0FORFNFUCB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY19udW1lcmljX3ZhbHVlc1tdID0gewogICAgICBMT0NBTEVfU0RFQ0lNQUwsCiAgICAgIExPQ0FMRV9TVEhPVVNBTkQsCiAgICAgIExPQ0FMRV9JRElHSVRTLAogICAgICBMT0NBTEVfSURJR0lUU1VCU1RJVFVUSU9OLAogICAgICBMT0NBTEVfU05BVElWRURJR0lUUywKICAgICAgTE9DQUxFX0lORUdOVU1CRVIsCiAgICAgIExPQ0FMRV9TTkVHQVRJVkVTSUdOLAogICAgICBMT0NBTEVfU1BPU0lUSVZFU0lHTiwKICAgICAgTE9DQUxFX1NHUk9VUElORyB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY190aW1lX3ZhbHVlc1tdID0gewogICAgICBMT0NBTEVfUzExNTksCiAgICAgIExPQ0FMRV9TMjM1OSwKICAgICAgTE9DQUxFX1NUSU1FLAogICAgICBMT0NBTEVfSVRJTUUsCiAgICAgIExPQ0FMRV9JVExaRVJPLAogICAgICBMT0NBTEVfU1NIT1JUREFURSwKICAgICAgTE9DQUxFX1NMT05HREFURSwKICAgICAgTE9DQUxFX1NEQVRFLAogICAgICBMT0NBTEVfSVRJTUVNQVJLUE9TTiwKICAgICAgTE9DQUxFX0lDQUxFTkRBUlRZUEUsCiAgICAgIExPQ0FMRV9JRklSU1REQVlPRldFRUssCiAgICAgIExPQ0FMRV9JRklSU1RXRUVLT0ZZRUFSLAogICAgICBMT0NBTEVfU1RJTUVGT1JNQVQsCiAgICAgIExPQ0FMRV9TWUVBUk1PTlRILAogICAgICBMT0NBTEVfSURBVEUgfTsKICAgIHN0YXRpYyBjb25zdCBMQ1RZUEUgbGNfbWVhc3VyZW1lbnRfdmFsdWVzW10gPSB7IExPQ0FMRV9JTUVBU1VSRSB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY190ZWxlcGhvbmVfdmFsdWVzW10gPSB7IExPQ0FMRV9JQ09VTlRSWSB9OwogICAgc3RhdGljIGNvbnN0IExDVFlQRSBsY19wYXBlcl92YWx1ZXNbXSA9IHsgTE9DQUxFX0lQQVBFUlNJWkUgfTsKCiAgICBVTklDT0RFX1NUUklORyBuYW1lVzsKICAgIFdDSEFSIGJ1ZmZlcldbODBdOwogICAgRFdPUkQgY291bnQsIGk7CiAgICBIQU5ETEUgaGtleTsKICAgIExDSUQgbGNpZCA9IEdldFVzZXJEZWZhdWx0TENJRCgpOwoKICAgIGlmICghKGhrZXkgPSBjcmVhdGVfcmVnaXN0cnlfa2V5KCkpKQogICAgICAgIHJldHVybjsgIC8qIGRvbid0IGRvIGFueXRoaW5nIGlmIHdlIGNhbid0IGNyZWF0ZSB0aGUgcmVnaXN0cnkga2V5ICovCgogICAgbG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbG9jYWxlVywgbGNpZF9MQ19NRVNTQUdFUywgbGNfbWVzc2FnZXNfdmFsdWVzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGxjX21lc3NhZ2VzX3ZhbHVlcykvc2l6ZW9mKGxjX21lc3NhZ2VzX3ZhbHVlc1swXSkgKTsKICAgIGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIGhrZXksIGxjX21vbmV0YXJ5VywgbGNpZF9MQ19NT05FVEFSWSwgbGNfbW9uZXRhcnlfdmFsdWVzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGxjX21vbmV0YXJ5X3ZhbHVlcykvc2l6ZW9mKGxjX21vbmV0YXJ5X3ZhbHVlc1swXSkgKTsKICAgIGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIGhrZXksIGxjX251bWVyaWNXLCBsY2lkX0xDX05VTUVSSUMsIGxjX251bWVyaWNfdmFsdWVzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2l6ZW9mKGxjX251bWVyaWNfdmFsdWVzKS9zaXplb2YobGNfbnVtZXJpY192YWx1ZXNbMF0pICk7CiAgICBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBoa2V5LCBsY190aW1lVywgbGNpZF9MQ19USU1FLCBsY190aW1lX3ZhbHVlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihsY190aW1lX3ZhbHVlcykvc2l6ZW9mKGxjX3RpbWVfdmFsdWVzWzBdKSApOwogICAgbG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbGNfbWVhc3VyZW1lbnRXLCBsY2lkX0xDX01FQVNVUkVNRU5ULCBsY19tZWFzdXJlbWVudF92YWx1ZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YobGNfbWVhc3VyZW1lbnRfdmFsdWVzKS9zaXplb2YobGNfbWVhc3VyZW1lbnRfdmFsdWVzWzBdKSApOwogICAgbG9jYWxlX3VwZGF0ZV9yZWdpc3RyeSggaGtleSwgbGNfdGVsZXBob25lVywgbGNpZF9MQ19URUxFUEhPTkUsIGxjX3RlbGVwaG9uZV92YWx1ZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YobGNfdGVsZXBob25lX3ZhbHVlcykvc2l6ZW9mKGxjX3RlbGVwaG9uZV92YWx1ZXNbMF0pICk7CiAgICBsb2NhbGVfdXBkYXRlX3JlZ2lzdHJ5KCBoa2V5LCBsY19wYXBlclcsIGxjaWRfTENfUEFQRVIsIGxjX3BhcGVyX3ZhbHVlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihsY19wYXBlcl92YWx1ZXMpL3NpemVvZihsY19wYXBlcl92YWx1ZXNbMF0pICk7CgogICAgaWYgKGxvY2FsZV91cGRhdGVfcmVnaXN0cnkoIGhrZXksIGxjX2N0eXBlVywgbGNpZF9MQ19DVFlQRSwgTlVMTCwgMCApKQogICAgewogICAgICAgIEhLRVkgbmxzX2tleSA9IE5MU19SZWdPcGVuS2V5KCAwLCBzekNvZGVwYWdlS2V5TmFtZSApOwoKICAgICAgICBmb3IgKGkgPSAwOyBpIDwgc2l6ZW9mKHVwZGF0ZV9jcF92YWx1ZXMpL3NpemVvZih1cGRhdGVfY3BfdmFsdWVzWzBdKTsgaSsrKQogICAgICAgIHsKICAgICAgICAgICAgY291bnQgPSBHZXRMb2NhbGVJbmZvVyggbGNpZCwgdXBkYXRlX2NwX3ZhbHVlc1tpXS52YWx1ZSB8IExPQ0FMRV9OT1VTRVJPVkVSUklERSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYnVmZmVyVywgc2l6ZW9mKGJ1ZmZlclcpL3NpemVvZihXQ0hBUikgKTsKICAgICAgICAgICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZuYW1lVywgdXBkYXRlX2NwX3ZhbHVlc1tpXS5uYW1lICk7CiAgICAgICAgICAgIE50U2V0VmFsdWVLZXkoIG5sc19rZXksICZuYW1lVywgMCwgUkVHX1NaLCBidWZmZXJXLCBjb3VudCAqIHNpemVvZihXQ0hBUikgKTsKICAgICAgICB9CiAgICAgICAgTnRDbG9zZSggbmxzX2tleSApOwogICAgfQoKICAgIE50Q2xvc2UoIGhrZXkgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgc2V0dXBfdW5peF9sb2NhbGVzCiAqLwpzdGF0aWMgVUlOVCBzZXR1cF91bml4X2xvY2FsZXModm9pZCkKewogICAgc3RydWN0IGxvY2FsZV9uYW1lIGxvY2FsZV9uYW1lOwogICAgV0NIQVIgYnVmZmVyWzEyOF0sIGN0eXBlX2J1ZmZbMTI4XTsKICAgIGNoYXIgKmxvY2FsZTsKICAgIFVJTlQgdW5peF9jcCA9IDA7CgogICAgaWYgKChsb2NhbGUgPSBzZXRsb2NhbGUoIExDX0NUWVBFLCBOVUxMICkpKQogICAgewogICAgICAgIHN0cmNweW5BdG9XKCBjdHlwZV9idWZmLCBsb2NhbGUsIHNpemVvZihjdHlwZV9idWZmKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgcGFyc2VfbG9jYWxlX25hbWUoIGN0eXBlX2J1ZmYsICZsb2NhbGVfbmFtZSApOwogICAgICAgIGxjaWRfTENfQ1RZUEUgPSBsb2NhbGVfbmFtZS5sY2lkOwogICAgICAgIHVuaXhfY3AgPSBsb2NhbGVfbmFtZS5jb2RlcGFnZTsKICAgIH0KICAgIGlmICghbGNpZF9MQ19DVFlQRSkgIC8qIHRoaXMgb25lIG5lZWRzIGEgZGVmYXVsdCB2YWx1ZSAqLwogICAgICAgIGxjaWRfTENfQ1RZUEUgPSBNQUtFTENJRCggTUFLRUxBTkdJRChMQU5HX0VOR0xJU0gsU1VCTEFOR19ERUZBVUxUKSwgU09SVF9ERUZBVUxUICk7CgogICAgVFJBQ0UoICJnb3QgbGNpZCAlMDR4ICglZCBtYXRjaGVzKSBmb3IgTENfQ1RZUEU9JXNcbiIsCiAgICAgICAgICAgbG9jYWxlX25hbWUubGNpZCwgbG9jYWxlX25hbWUubWF0Y2hlcywgZGVidWdzdHJfYShsb2NhbGUpICk7CgojZGVmaW5lIEdFVF9VTklYX0xPQ0FMRShjYXQpIGRvIFwKICAgIGlmICgobG9jYWxlID0gc2V0bG9jYWxlKCBjYXQsIE5VTEwgKSkpIFwKICAgIHsgXAogICAgICAgIHN0cmNweW5BdG9XKCBidWZmZXIsIGxvY2FsZSwgc2l6ZW9mKGJ1ZmZlcikvc2l6ZW9mKFdDSEFSKSApOyBcCiAgICAgICAgaWYgKCFzdHJjbXBXKCBidWZmZXIsIGN0eXBlX2J1ZmYgKSkgbGNpZF8jI2NhdCA9IGxjaWRfTENfQ1RZUEU7IFwKICAgICAgICBlbHNlIHsgXAogICAgICAgICAgICBwYXJzZV9sb2NhbGVfbmFtZSggYnVmZmVyLCAmbG9jYWxlX25hbWUgKTsgIFwKICAgICAgICAgICAgbGNpZF8jI2NhdCA9IGxvY2FsZV9uYW1lLmxjaWQ7IFwKICAgICAgICAgICAgVFJBQ0UoICJnb3QgbGNpZCAlMDR4ICglZCBtYXRjaGVzKSBmb3IgIiAjY2F0ICI9JXNcbiIsICAgICAgICBcCiAgICAgICAgICAgICAgICAgICBsb2NhbGVfbmFtZS5sY2lkLCBsb2NhbGVfbmFtZS5tYXRjaGVzLCBkZWJ1Z3N0cl9hKGxvY2FsZSkgKTsgXAogICAgICAgIH0gXAogICAgfSB3aGlsZSAoMCkKCiAgICBHRVRfVU5JWF9MT0NBTEUoIExDX0NPTExBVEUgKTsKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfTUVTU0FHRVMgKTsKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfTU9ORVRBUlkgKTsKICAgIEdFVF9VTklYX0xPQ0FMRSggTENfTlVNRVJJQyApOwogICAgR0VUX1VOSVhfTE9DQUxFKCBMQ19USU1FICk7CiNpZmRlZiBMQ19QQVBFUgogICAgR0VUX1VOSVhfTE9DQUxFKCBMQ19QQVBFUiApOwojZW5kaWYKI2lmZGVmIExDX01FQVNVUkVNRU5UCiAgICBHRVRfVU5JWF9MT0NBTEUoIExDX01FQVNVUkVNRU5UICk7CiNlbmRpZgojaWZkZWYgTENfVEVMRVBIT05FCiAgICBHRVRfVU5JWF9MT0NBTEUoIExDX1RFTEVQSE9ORSApOwojZW5kaWYKCiN1bmRlZiBHRVRfVU5JWF9MT0NBTEUKCiAgICByZXR1cm4gdW5peF9jcDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0VXNlckRlZmF1bHRMYW5nSUQgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCBsYW5ndWFnZSBJZCBmb3IgdGhlIGN1cnJlbnQgdXNlci4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQU5HSUQgb2YgdGhlIGRlZmF1bHQgbGFuZ3VhZ2UgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqLwpMQU5HSUQgV0lOQVBJIEdldFVzZXJEZWZhdWx0TGFuZ0lEKHZvaWQpCnsKICAgIHJldHVybiBMQU5HSURGUk9NTENJRChHZXRVc2VyRGVmYXVsdExDSUQoKSk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUdldFN5c3RlbURlZmF1bHRMYW5nSUQgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCBsYW5ndWFnZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQU5HSUQgb2YgdGhlIGRlZmF1bHQgbGFuZ3VhZ2UgZm9yIHRoZSBzeXN0ZW0uCiAqLwpMQU5HSUQgV0lOQVBJIEdldFN5c3RlbURlZmF1bHRMYW5nSUQodm9pZCkKewogICAgcmV0dXJuIExBTkdJREZST01MQ0lEKEdldFN5c3RlbURlZmF1bHRMQ0lEKCkpOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlHZXRVc2VyRGVmYXVsdExDSUQgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCBsb2NhbGUgSWQgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgVGhlIGN1cnJlbnQgTENJRCBvZiB0aGUgZGVmYXVsdCBsb2NhbGUgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqLwpMQ0lEIFdJTkFQSSBHZXRVc2VyRGVmYXVsdExDSUQodm9pZCkKewogICAgTENJRCBsY2lkOwogICAgTnRRdWVyeURlZmF1bHRMb2NhbGUoIFRSVUUsICZsY2lkICk7CiAgICByZXR1cm4gbGNpZDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0U3lzdGVtRGVmYXVsdExDSUQgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCBsb2NhbGUgSWQgZm9yIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgVGhlIGN1cnJlbnQgTENJRCBvZiB0aGUgZGVmYXVsdCBsb2NhbGUgZm9yIHRoZSBzeXN0ZW0uCiAqLwpMQ0lEIFdJTkFQSSBHZXRTeXN0ZW1EZWZhdWx0TENJRCh2b2lkKQp7CiAgICBMQ0lEIGxjaWQ7CiAgICBOdFF1ZXJ5RGVmYXVsdExvY2FsZSggRkFMU0UsICZsY2lkICk7CiAgICByZXR1cm4gbGNpZDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0VXNlckRlZmF1bHRVSUxhbmd1YWdlIChLRVJORUwzMi5AKQogKgogKiBHZXQgdGhlIGRlZmF1bHQgdXNlciBpbnRlcmZhY2UgbGFuZ3VhZ2UgSWQgZm9yIHRoZSBjdXJyZW50IHVzZXIuCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgVGhlIGN1cnJlbnQgTEFOR0lEIG9mIHRoZSBkZWZhdWx0IFVJIGxhbmd1YWdlIGZvciB0aGUgY3VycmVudCB1c2VyLgogKi8KTEFOR0lEIFdJTkFQSSBHZXRVc2VyRGVmYXVsdFVJTGFuZ3VhZ2Uodm9pZCkKewogICAgTEFOR0lEIGxhbmc7CiAgICBOdFF1ZXJ5RGVmYXVsdFVJTGFuZ3VhZ2UoICZsYW5nICk7CiAgICByZXR1cm4gbGFuZzsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJR2V0U3lzdGVtRGVmYXVsdFVJTGFuZ3VhZ2UgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgZGVmYXVsdCB1c2VyIGludGVyZmFjZSBsYW5ndWFnZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBOb25lLgogKgogKiBSRVRVUk5TCiAqICBUaGUgY3VycmVudCBMQU5HSUQgb2YgdGhlIGRlZmF1bHQgVUkgbGFuZ3VhZ2UgZm9yIHRoZSBzeXN0ZW0uIFRoaXMgaXMKICogIHR5cGljYWxseSB0aGUgc2FtZSBsYW5ndWFnZSB1c2VkIGR1cmluZyB0aGUgaW5zdGFsbGF0aW9uIHByb2Nlc3MuCiAqLwpMQU5HSUQgV0lOQVBJIEdldFN5c3RlbURlZmF1bHRVSUxhbmd1YWdlKHZvaWQpCnsKICAgIExBTkdJRCBsYW5nOwogICAgTnRRdWVyeUluc3RhbGxVSUxhbmd1YWdlKCAmbGFuZyApOwogICAgcmV0dXJuIGxhbmc7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIExvY2FsZU5hbWVUb0xDSUQgIChLRVJORUwzMi5AKQogKi8KTENJRCBXSU5BUEkgTG9jYWxlTmFtZVRvTENJRCggTFBDV1NUUiBuYW1lLCBEV09SRCBmbGFncyApCnsKICAgIHN0cnVjdCBsb2NhbGVfbmFtZSBsb2NhbGVfbmFtZTsKCiAgICBpZiAoZmxhZ3MpIEZJWE1FKCAidW5zdXBwb3J0ZWQgZmxhZ3MgJXhcbiIsIGZsYWdzICk7CgogICAgcGFyc2VfbG9jYWxlX25hbWUoIG5hbWUsICZsb2NhbGVfbmFtZSApOwoKICAgIFRSQUNFKCAiZm91bmQgbGNpZCAleCBmb3IgJXMsIG1hdGNoZXMgJWRcbiIsCiAgICAgICAgICAgbG9jYWxlX25hbWUubGNpZCwgZGVidWdzdHJfdyhuYW1lKSwgbG9jYWxlX25hbWUubWF0Y2hlcyApOwoKICAgIGlmICghbG9jYWxlX25hbWUubWF0Y2hlcykKICAgICAgICBXQVJOKCAibG9jYWxlICVzIG5vdCByZWNvZ25pemVkLCBkZWZhdWx0aW5nIHRvIEVuZ2xpc2hcbiIsIGRlYnVnc3RyX3cobmFtZSkgKTsKICAgIGVsc2UgaWYgKGxvY2FsZV9uYW1lLm1hdGNoZXMgPT0gMSkKICAgICAgICBXQVJOKCAibG9jYWxlICVzIG5vdCByZWNvZ25pemVkLCBkZWZhdWx0aW5nIHRvICVzXG4iLAogICAgICAgICAgICAgIGRlYnVnc3RyX3cobmFtZSksIGRlYnVnc3RyX3cobG9jYWxlX25hbWUubGFuZykgKTsKCiAgICByZXR1cm4gbG9jYWxlX25hbWUubGNpZDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgTENJRFRvTG9jYWxlTmFtZSAgKEtFUk5FTDMyLkApCiAqLwpJTlQgV0lOQVBJIExDSURUb0xvY2FsZU5hbWUoIExDSUQgbGNpZCwgTFBXU1RSIG5hbWUsIElOVCBjb3VudCwgRFdPUkQgZmxhZ3MgKQp7CiAgICBpZiAoZmxhZ3MpIEZJWE1FKCAidW5zdXBwb3J0ZWQgZmxhZ3MgJXhcbiIsIGZsYWdzICk7CgogICAgcmV0dXJuIEdldExvY2FsZUluZm9XKCBsY2lkLCBMT0NBTEVfU05BTUUgfCBMT0NBTEVfTk9VU0VST1ZFUlJJREUsIG5hbWUsIGNvdW50ICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlnZXRfbG9jYWxlX3ZhbHVlX25hbWUKICoKICogR2V0cyB0aGUgcmVnaXN0cnkgdmFsdWUgbmFtZSBmb3IgYSBnaXZlbiBsY3R5cGUuCiAqLwpzdGF0aWMgY29uc3QgV0NIQVIgKmdldF9sb2NhbGVfdmFsdWVfbmFtZSggRFdPUkQgbGN0eXBlICkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlDYWxlbmRhclR5cGVXW10gPSB7J2knLCdDJywnYScsJ2wnLCdlJywnbicsJ2QnLCdhJywncicsJ1QnLCd5JywncCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlDb3VudHJ5V1tdID0geydpJywnQycsJ28nLCd1JywnbicsJ3QnLCdyJywneScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUN1cnJEaWdpdHNXW10gPSB7J2knLCdDJywndScsJ3InLCdyJywnRCcsJ2knLCdnJywnaScsJ3QnLCdzJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpQ3VycmVuY3lXW10gPSB7J2knLCdDJywndScsJ3InLCdyJywnZScsJ24nLCdjJywneScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaURhdGVXW10gPSB7J2knLCdEJywnYScsJ3QnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRGlnaXRzV1tdID0geydpJywnRCcsJ2knLCdnJywnaScsJ3QnLCdzJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpRmlyc3REYXlPZldlZWtXW10gPSB7J2knLCdGJywnaScsJ3InLCdzJywndCcsJ0QnLCdhJywneScsJ08nLCdmJywnVycsJ2UnLCdlJywnaycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUZpcnN0V2Vla09mWWVhcldbXSA9IHsnaScsJ0YnLCdpJywncicsJ3MnLCd0JywnVycsJ2UnLCdlJywnaycsJ08nLCdmJywnWScsJ2UnLCdhJywncicsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaUxEYXRlV1tdID0geydpJywnTCcsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlMWmVyb1dbXSA9IHsnaScsJ0wnLCdaJywnZScsJ3InLCdvJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpTWVhc3VyZVdbXSA9IHsnaScsJ00nLCdlJywnYScsJ3MnLCd1JywncicsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIGlOZWdDdXJyV1tdID0geydpJywnTicsJ2UnLCdnJywnQycsJ3UnLCdyJywncicsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaU5lZ051bWJlcldbXSA9IHsnaScsJ04nLCdlJywnZycsJ04nLCd1JywnbScsJ2InLCdlJywncicsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaVBhcGVyU2l6ZVdbXSA9IHsnaScsJ1AnLCdhJywncCcsJ2UnLCdyJywnUycsJ2knLCd6JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaVRMWmVyb1dbXSA9IHsnaScsJ1QnLCdMJywnWicsJ2UnLCdyJywnbycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgaVRpbWVQcmVmaXhXW10gPSB7J2knLCdUJywnaScsJ20nLCdlJywnUCcsJ3InLCdlJywnZicsJ2knLCd4JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpVGltZVdbXSA9IHsnaScsJ1QnLCdpJywnbScsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHMxMTU5V1tdID0geydzJywnMScsJzEnLCc1JywnOScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgczIzNTlXW10gPSB7J3MnLCcyJywnMycsJzUnLCc5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzQ291bnRyeVdbXSA9IHsncycsJ0MnLCdvJywndScsJ24nLCd0JywncicsJ3knLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNDdXJyZW5jeVdbXSA9IHsncycsJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ2MnLCd5JywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzRGF0ZVdbXSA9IHsncycsJ0QnLCdhJywndCcsJ2UnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNEZWNpbWFsV1tdID0geydzJywnRCcsJ2UnLCdjJywnaScsJ20nLCdhJywnbCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc0dyb3VwaW5nV1tdID0geydzJywnRycsJ3InLCdvJywndScsJ3AnLCdpJywnbicsJ2cnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNMYW5ndWFnZVdbXSA9IHsncycsJ0wnLCdhJywnbicsJ2cnLCd1JywnYScsJ2cnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTGlzdFdbXSA9IHsncycsJ0wnLCdpJywncycsJ3QnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNMb25nRGF0ZVdbXSA9IHsncycsJ0wnLCdvJywnbicsJ2cnLCdEJywnYScsJ3QnLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTW9uRGVjaW1hbFNlcFdbXSA9IHsncycsJ00nLCdvJywnbicsJ0QnLCdlJywnYycsJ2knLCdtJywnYScsJ2wnLCdTJywnZScsJ3AnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNNb25Hcm91cGluZ1dbXSA9IHsncycsJ00nLCdvJywnbicsJ0cnLCdyJywnbycsJ3UnLCdwJywnaScsJ24nLCdnJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzTW9uVGhvdXNhbmRTZXBXW10gPSB7J3MnLCdNJywnbycsJ24nLCdUJywnaCcsJ28nLCd1JywncycsJ2EnLCduJywnZCcsJ1MnLCdlJywncCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc05hdGl2ZURpZ2l0c1dbXSA9IHsncycsJ04nLCdhJywndCcsJ2knLCd2JywnZScsJ0QnLCdpJywnZycsJ2knLCd0JywncycsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc05lZ2F0aXZlU2lnbldbXSA9IHsncycsJ04nLCdlJywnZycsJ2EnLCd0JywnaScsJ3YnLCdlJywnUycsJ2knLCdnJywnbicsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1Bvc2l0aXZlU2lnbldbXSA9IHsncycsJ1AnLCdvJywncycsJ2knLCd0JywnaScsJ3YnLCdlJywnUycsJ2knLCdnJywnbicsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1Nob3J0RGF0ZVdbXSA9IHsncycsJ1MnLCdoJywnbycsJ3InLCd0JywnRCcsJ2EnLCd0JywnZScsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1Rob3VzYW5kV1tdID0geydzJywnVCcsJ2gnLCdvJywndScsJ3MnLCdhJywnbicsJ2QnLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIHNUaW1lRm9ybWF0V1tdID0geydzJywnVCcsJ2knLCdtJywnZScsJ0YnLCdvJywncicsJ20nLCdhJywndCcsMH07CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgc1RpbWVXW10gPSB7J3MnLCdUJywnaScsJ20nLCdlJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzWWVhck1vbnRoV1tdID0geydzJywnWScsJ2UnLCdhJywncicsJ00nLCdvJywnbicsJ3QnLCdoJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBOdW1TaGFwZVdbXSA9IHsnTicsJ3UnLCdtJywncycsJ2gnLCdhJywncCcsJ2UnLDB9OwoKICAgIHN3aXRjaCAobGN0eXBlKQogICAgewogICAgLyogVGhlc2UgdmFsdWVzIGFyZSB1c2VkIGJ5IFNldExvY2FsZUluZm8gYW5kIEdldExvY2FsZUluZm8sIGFuZAogICAgICogdGhlIHZhbHVlcyBhcmUgc3RvcmVkIGluIHRoZSByZWdpc3RyeSwgY29uZmlybWVkIHVuZGVyIFdpbmRvd3MuCiAgICAgKi8KICAgIGNhc2UgTE9DQUxFX0lDQUxFTkRBUlRZUEU6ICAgIHJldHVybiBpQ2FsZW5kYXJUeXBlVzsKICAgIGNhc2UgTE9DQUxFX0lDVVJSRElHSVRTOiAgICAgIHJldHVybiBpQ3VyckRpZ2l0c1c7CiAgICBjYXNlIExPQ0FMRV9JQ1VSUkVOQ1k6ICAgICAgICByZXR1cm4gaUN1cnJlbmN5VzsKICAgIGNhc2UgTE9DQUxFX0lESUdJVFM6ICAgICAgICAgIHJldHVybiBpRGlnaXRzVzsKICAgIGNhc2UgTE9DQUxFX0lGSVJTVERBWU9GV0VFSzogIHJldHVybiBpRmlyc3REYXlPZldlZWtXOwogICAgY2FzZSBMT0NBTEVfSUZJUlNUV0VFS09GWUVBUjogcmV0dXJuIGlGaXJzdFdlZWtPZlllYXJXOwogICAgY2FzZSBMT0NBTEVfSUxaRVJPOiAgICAgICAgICAgcmV0dXJuIGlMWmVyb1c7CiAgICBjYXNlIExPQ0FMRV9JTUVBU1VSRTogICAgICAgICByZXR1cm4gaU1lYXN1cmVXOwogICAgY2FzZSBMT0NBTEVfSU5FR0NVUlI6ICAgICAgICAgcmV0dXJuIGlOZWdDdXJyVzsKICAgIGNhc2UgTE9DQUxFX0lORUdOVU1CRVI6ICAgICAgIHJldHVybiBpTmVnTnVtYmVyVzsKICAgIGNhc2UgTE9DQUxFX0lQQVBFUlNJWkU6ICAgICAgIHJldHVybiBpUGFwZXJTaXplVzsKICAgIGNhc2UgTE9DQUxFX0lUSU1FOiAgICAgICAgICAgIHJldHVybiBpVGltZVc7CiAgICBjYXNlIExPQ0FMRV9TMTE1OTogICAgICAgICAgICByZXR1cm4gczExNTlXOwogICAgY2FzZSBMT0NBTEVfUzIzNTk6ICAgICAgICAgICAgcmV0dXJuIHMyMzU5VzsKICAgIGNhc2UgTE9DQUxFX1NDVVJSRU5DWTogICAgICAgIHJldHVybiBzQ3VycmVuY3lXOwogICAgY2FzZSBMT0NBTEVfU0RBVEU6ICAgICAgICAgICAgcmV0dXJuIHNEYXRlVzsKICAgIGNhc2UgTE9DQUxFX1NERUNJTUFMOiAgICAgICAgIHJldHVybiBzRGVjaW1hbFc7CiAgICBjYXNlIExPQ0FMRV9TR1JPVVBJTkc6ICAgICAgICByZXR1cm4gc0dyb3VwaW5nVzsKICAgIGNhc2UgTE9DQUxFX1NMSVNUOiAgICAgICAgICAgIHJldHVybiBzTGlzdFc7CiAgICBjYXNlIExPQ0FMRV9TTE9OR0RBVEU6ICAgICAgICByZXR1cm4gc0xvbmdEYXRlVzsKICAgIGNhc2UgTE9DQUxFX1NNT05ERUNJTUFMU0VQOiAgIHJldHVybiBzTW9uRGVjaW1hbFNlcFc7CiAgICBjYXNlIExPQ0FMRV9TTU9OR1JPVVBJTkc6ICAgICByZXR1cm4gc01vbkdyb3VwaW5nVzsKICAgIGNhc2UgTE9DQUxFX1NNT05USE9VU0FORFNFUDogIHJldHVybiBzTW9uVGhvdXNhbmRTZXBXOwogICAgY2FzZSBMT0NBTEVfU05FR0FUSVZFU0lHTjogICAgcmV0dXJuIHNOZWdhdGl2ZVNpZ25XOwogICAgY2FzZSBMT0NBTEVfU1BPU0lUSVZFU0lHTjogICAgcmV0dXJuIHNQb3NpdGl2ZVNpZ25XOwogICAgY2FzZSBMT0NBTEVfU1NIT1JUREFURTogICAgICAgcmV0dXJuIHNTaG9ydERhdGVXOwogICAgY2FzZSBMT0NBTEVfU1RIT1VTQU5EOiAgICAgICAgcmV0dXJuIHNUaG91c2FuZFc7CiAgICBjYXNlIExPQ0FMRV9TVElNRTogICAgICAgICAgICByZXR1cm4gc1RpbWVXOwogICAgY2FzZSBMT0NBTEVfU1RJTUVGT1JNQVQ6ICAgICAgcmV0dXJuIHNUaW1lRm9ybWF0VzsKICAgIGNhc2UgTE9DQUxFX1NZRUFSTU9OVEg6ICAgICAgIHJldHVybiBzWWVhck1vbnRoVzsKCiAgICAvKiBUaGUgZm9sbG93aW5nIGFyZSBub3QgbGlzdGVkIHVuZGVyIE1TRE4gYXMgc3VwcG9ydGVkLAogICAgICogYnV0IHNlZW0gdG8gYmUgdXNlZCBhbmQgYWxzbyBzdG9yZWQgaW4gdGhlIHJlZ2lzdHJ5LgogICAgICovCiAgICBjYXNlIExPQ0FMRV9JQ09VTlRSWTogICAgICAgICByZXR1cm4gaUNvdW50cnlXOwogICAgY2FzZSBMT0NBTEVfSURBVEU6ICAgICAgICAgICAgcmV0dXJuIGlEYXRlVzsKICAgIGNhc2UgTE9DQUxFX0lMREFURTogICAgICAgICAgIHJldHVybiBpTERhdGVXOwogICAgY2FzZSBMT0NBTEVfSVRMWkVSTzogICAgICAgICAgcmV0dXJuIGlUTFplcm9XOwogICAgY2FzZSBMT0NBTEVfU0NPVU5UUlk6ICAgICAgICAgcmV0dXJuIHNDb3VudHJ5VzsKICAgIGNhc2UgTE9DQUxFX1NMQU5HVUFHRTogICAgICAgIHJldHVybiBzTGFuZ3VhZ2VXOwoKICAgIC8qIFRoZSBmb2xsb3dpbmcgYXJlIHVzZWQgaW4gWFAgYW5kIGxhdGVyICovCiAgICBjYXNlIExPQ0FMRV9JRElHSVRTVUJTVElUVVRJT046IHJldHVybiBOdW1TaGFwZVc7CiAgICBjYXNlIExPQ0FMRV9TTkFUSVZFRElHSVRTOiAgICAgIHJldHVybiBzTmF0aXZlRGlnaXRzVzsKICAgIGNhc2UgTE9DQUxFX0lUSU1FTUFSS1BPU046ICAgICAgcmV0dXJuIGlUaW1lUHJlZml4VzsKICAgIH0KICAgIHJldHVybiBOVUxMOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJZ2V0X3JlZ2lzdHJ5X2xvY2FsZV9pbmZvCiAqCiAqIFJldHJpZXZlIHVzZXItbW9kaWZpZWQgbG9jYWxlIGluZm8gZnJvbSB0aGUgcmVnaXN0cnkuCiAqIFJldHVybiBsZW5ndGgsIDAgb24gZXJyb3IsIC0xIGlmIG5vdCBmb3VuZC4KICovCnN0YXRpYyBJTlQgZ2V0X3JlZ2lzdHJ5X2xvY2FsZV9pbmZvKCBMUENXU1RSIHZhbHVlLCBMUFdTVFIgYnVmZmVyLCBJTlQgbGVuICkKewogICAgRFdPUkQgc2l6ZTsKICAgIElOVCByZXQ7CiAgICBIQU5ETEUgaGtleTsKICAgIE5UU1RBVFVTIHN0YXR1czsKICAgIFVOSUNPREVfU1RSSU5HIG5hbWVXOwogICAgS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKmluZm87CiAgICBzdGF0aWMgY29uc3QgaW50IGluZm9fc2l6ZSA9IEZJRUxEX09GRlNFVChLRVlfVkFMVUVfUEFSVElBTF9JTkZPUk1BVElPTiwgRGF0YSk7CgogICAgaWYgKCEoaGtleSA9IGNyZWF0ZV9yZWdpc3RyeV9rZXkoKSkpIHJldHVybiAtMTsKCiAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJm5hbWVXLCB2YWx1ZSApOwogICAgc2l6ZSA9IGluZm9fc2l6ZSArIGxlbiAqIHNpemVvZihXQ0hBUik7CgogICAgaWYgKCEoaW5mbyA9IEhlYXBBbGxvYyggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgc2l6ZSApKSkKICAgIHsKICAgICAgICBOdENsb3NlKCBoa2V5ICk7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIHN0YXR1cyA9IE50UXVlcnlWYWx1ZUtleSggaGtleSwgJm5hbWVXLCBLZXlWYWx1ZVBhcnRpYWxJbmZvcm1hdGlvbiwgaW5mbywgc2l6ZSwgJnNpemUgKTsKCiAgICBpZiAoIXN0YXR1cykKICAgIHsKICAgICAgICByZXQgPSAoc2l6ZSAtIGluZm9fc2l6ZSkgLyBzaXplb2YoV0NIQVIpOwogICAgICAgIC8qIGFwcGVuZCB0ZXJtaW5hdGluZyBudWxsIGlmIG5lZWRlZCAqLwogICAgICAgIGlmICghcmV0IHx8ICgoV0NIQVIgKilpbmZvLT5EYXRhKVtyZXQtMV0pCiAgICAgICAgewogICAgICAgICAgICBpZiAocmV0IDwgbGVuIHx8ICFidWZmZXIpIHJldCsrOwogICAgICAgICAgICBlbHNlCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUiApOwogICAgICAgICAgICAgICAgcmV0ID0gMDsKICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBpZiAocmV0ICYmIGJ1ZmZlcikKICAgICAgICB7CiAgICAgICAgICAgIG1lbWNweSggYnVmZmVyLCBpbmZvLT5EYXRhLCAocmV0LTEpICogc2l6ZW9mKFdDSEFSKSApOwogICAgICAgICAgICBidWZmZXJbcmV0LTFdID0gMDsKICAgICAgICB9CiAgICB9CiAgICBlbHNlIGlmIChzdGF0dXMgPT0gU1RBVFVTX0JVRkZFUl9PVkVSRkxPVyAmJiAhYnVmZmVyKQogICAgewogICAgICAgIHJldCA9IChzaXplIC0gaW5mb19zaXplKSAvIHNpemVvZihXQ0hBUikgKyAxOwogICAgfQogICAgZWxzZSBpZiAoc3RhdHVzID09IFNUQVRVU19PQkpFQ1RfTkFNRV9OT1RfRk9VTkQpCiAgICB7CiAgICAgICAgcmV0ID0gLTE7CiAgICB9CiAgICBlbHNlCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBSdGxOdFN0YXR1c1RvRG9zRXJyb3Ioc3RhdHVzKSApOwogICAgICAgIHJldCA9IDA7CiAgICB9CiAgICBOdENsb3NlKCBoa2V5ICk7CiAgICBIZWFwRnJlZSggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgaW5mbyApOwogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUdldExvY2FsZUluZm9BIChLRVJORUwzMi5AKQogKgogKiBHZXQgaW5mb3JtYXRpb24gYWJvdXQgYW4gYXNwZWN0IG9mIGEgbG9jYWxlLgogKgogKiBQQVJBTVMKICogIGxjaWQgICBbSV0gTENJRCBvZiB0aGUgbG9jYWxlCiAqICBsY3R5cGUgW0ldIExDVFlQRV8gZmxhZ3MgZnJvbSAid2lubmxzLmgiCiAqICBidWZmZXIgW09dIERlc3RpbmF0aW9uIGZvciB0aGUgaW5mb3JtYXRpb24KICogIGxlbiAgICBbSV0gTGVuZ3RoIG9mIGJ1ZmZlciBpbiBjaGFyYWN0ZXJzCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRoZSBzaXplIG9mIHRoZSBkYXRhIHJlcXVlc3RlZC4gSWYgYnVmZmVyIGlzIG5vbi1OVUxMLCBpdCBpcyBmaWxsZWQKICogICAgICAgICAgIHdpdGggdGhlIGluZm9ybWF0aW9uLgogKiAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqCiAqIE5PVEVTCiAqICAtIExPQ0FMRV9ORVVUUkFMIGlzIGVxdWFsIHRvIExPQ0FMRV9TWVNURU1fREVGQVVMVAogKiAgLSBUaGUgc3RyaW5nIHJldHVybmVkIGlzIE5VTCB0ZXJtaW5hdGVkLCBleGNlcHQgZm9yIExPQ0FMRV9GT05UU0lHTkFUVVJFLAogKiAgICB3aGljaCBpcyBhIGJpdCBzdHJpbmcuCiAqLwpJTlQgV0lOQVBJIEdldExvY2FsZUluZm9BKCBMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUsIExQU1RSIGJ1ZmZlciwgSU5UIGxlbiApCnsKICAgIFdDSEFSICpidWZmZXJXOwogICAgSU5UIGxlblcsIHJldDsKCiAgICBUUkFDRSggIihsY2lkPTB4JXgsbGN0eXBlPTB4JXgsJXAsJWQpXG4iLCBsY2lkLCBsY3R5cGUsIGJ1ZmZlciwgbGVuICk7CgogICAgaWYgKGxlbiA8IDAgfHwgKGxlbiAmJiAhYnVmZmVyKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CiAgICBpZiAoIWxlbikgYnVmZmVyID0gTlVMTDsKCiAgICBpZiAoIShsZW5XID0gR2V0TG9jYWxlSW5mb1coIGxjaWQsIGxjdHlwZSwgTlVMTCwgMCApKSkgcmV0dXJuIDA7CgogICAgaWYgKCEoYnVmZmVyVyA9IEhlYXBBbGxvYyggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGVuVyAqIHNpemVvZihXQ0hBUikgKSkpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSApOwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgaWYgKChyZXQgPSBHZXRMb2NhbGVJbmZvVyggbGNpZCwgbGN0eXBlLCBidWZmZXJXLCBsZW5XICkpKQogICAgewogICAgICAgIGlmICgobGN0eXBlICYgTE9DQUxFX1JFVFVSTl9OVU1CRVIpIHx8CiAgICAgICAgICAgICgobGN0eXBlICYgfkxPQ0FMRV9MT0NBTEVJTkZPRkxBR1NNQVNLKSA9PSBMT0NBTEVfRk9OVFNJR05BVFVSRSkpCiAgICAgICAgewogICAgICAgICAgICAvKiBpdCdzIG5vdCBhbiBBU0NJSSBzdHJpbmcsIGp1c3QgYnl0ZXMgKi8KICAgICAgICAgICAgcmV0ICo9IHNpemVvZihXQ0hBUik7CiAgICAgICAgICAgIGlmIChidWZmZXIpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIGlmIChyZXQgPD0gbGVuKSBtZW1jcHkoIGJ1ZmZlciwgYnVmZmVyVywgcmV0ICk7CiAgICAgICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSICk7CiAgICAgICAgICAgICAgICAgICAgcmV0ID0gMDsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICBVSU5UIGNvZGVwYWdlID0gQ1BfQUNQOwogICAgICAgICAgICBpZiAoIShsY3R5cGUgJiBMT0NBTEVfVVNFX0NQX0FDUCkpIGNvZGVwYWdlID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKICAgICAgICAgICAgcmV0ID0gV2lkZUNoYXJUb011bHRpQnl0ZSggY29kZXBhZ2UsIDAsIGJ1ZmZlclcsIHJldCwgYnVmZmVyLCBsZW4sIE5VTEwsIE5VTEwgKTsKICAgICAgICB9CiAgICB9CiAgICBIZWFwRnJlZSggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgYnVmZmVyVyApOwogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUdldExvY2FsZUluZm9XIChLRVJORUwzMi5AKQogKgogKiBTZWUgR2V0TG9jYWxlSW5mb0EuCiAqLwpJTlQgV0lOQVBJIEdldExvY2FsZUluZm9XKCBMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUsIExQV1NUUiBidWZmZXIsIElOVCBsZW4gKQp7CiAgICBMQU5HSUQgbGFuZ19pZDsKICAgIEhSU1JDIGhyc3JjOwogICAgSEdMT0JBTCBobWVtOwogICAgSU5UIHJldDsKICAgIFVJTlQgbGNmbGFnczsKICAgIGNvbnN0IFdDSEFSICpwOwogICAgdW5zaWduZWQgaW50IGk7CgogICAgaWYgKGxlbiA8IDAgfHwgKGxlbiAmJiAhYnVmZmVyKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CiAgICBpZiAoIWxlbikgYnVmZmVyID0gTlVMTDsKCiAgICBsY2lkID0gY29udmVydF9kZWZhdWx0X2xjaWQoIGxjaWQsIGxjdHlwZSApOwoKICAgIGxjZmxhZ3MgPSBsY3R5cGUgJiBMT0NBTEVfTE9DQUxFSU5GT0ZMQUdTTUFTSzsKICAgIGxjdHlwZSAmPSAweGZmZmY7CgogICAgVFJBQ0UoICIobGNpZD0weCV4LGxjdHlwZT0weCV4LCVwLCVkKVxuIiwgbGNpZCwgbGN0eXBlLCBidWZmZXIsIGxlbiApOwoKICAgIC8qIGZpcnN0IGNoZWNrIGZvciBvdmVycmlkZXMgaW4gdGhlIHJlZ2lzdHJ5ICovCgogICAgaWYgKCEobGNmbGFncyAmIExPQ0FMRV9OT1VTRVJPVkVSUklERSkgJiYKICAgICAgICBsY2lkID09IGNvbnZlcnRfZGVmYXVsdF9sY2lkKCBMT0NBTEVfVVNFUl9ERUZBVUxULCBsY3R5cGUgKSkKICAgIHsKICAgICAgICBjb25zdCBXQ0hBUiAqdmFsdWUgPSBnZXRfbG9jYWxlX3ZhbHVlX25hbWUobGN0eXBlKTsKCiAgICAgICAgaWYgKHZhbHVlKQogICAgICAgIHsKICAgICAgICAgICAgaWYgKGxjZmxhZ3MgJiBMT0NBTEVfUkVUVVJOX05VTUJFUikKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NIQVIgdG1wWzE2XTsKICAgICAgICAgICAgICAgIHJldCA9IGdldF9yZWdpc3RyeV9sb2NhbGVfaW5mbyggdmFsdWUsIHRtcCwgc2l6ZW9mKHRtcCkvc2l6ZW9mKFdDSEFSKSApOwogICAgICAgICAgICAgICAgaWYgKHJldCA+IDApCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgV0NIQVIgKmVuZDsKICAgICAgICAgICAgICAgICAgICBVSU5UIG51bWJlciA9IHN0cnRvbFcoIHRtcCwgJmVuZCwgMTAgKTsKICAgICAgICAgICAgICAgICAgICBpZiAoKmVuZCkgIC8qIGludmFsaWQgbnVtYmVyICovCiAgICAgICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfRkxBR1MgKTsKICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgIHJldCA9IHNpemVvZihVSU5UKS9zaXplb2YoV0NIQVIpOwogICAgICAgICAgICAgICAgICAgIGlmICghYnVmZmVyKSByZXR1cm4gcmV0OwogICAgICAgICAgICAgICAgICAgIGlmIChyZXQgPiBsZW4pCiAgICAgICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsKICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgIG1lbWNweSggYnVmZmVyLCAmbnVtYmVyLCBzaXplb2YobnVtYmVyKSApOwogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgICAgIGVsc2UgcmV0ID0gZ2V0X3JlZ2lzdHJ5X2xvY2FsZV9pbmZvKCB2YWx1ZSwgYnVmZmVyLCBsZW4gKTsKCiAgICAgICAgICAgIGlmIChyZXQgIT0gLTEpIHJldHVybiByZXQ7CiAgICAgICAgfQogICAgfQoKICAgIC8qIG5vdyBsb2FkIGl0IGZyb20ga2VybmVsIHJlc291cmNlcyAqLwoKICAgIGxhbmdfaWQgPSBMQU5HSURGUk9NTENJRCggbGNpZCApOwoKICAgIC8qIHJlcGxhY2UgU1VCTEFOR19ORVVUUkFMIGJ5IFNVQkxBTkdfREVGQVVMVCAqLwogICAgaWYgKFNVQkxBTkdJRChsYW5nX2lkKSA9PSBTVUJMQU5HX05FVVRSQUwpCiAgICAgICAgbGFuZ19pZCA9IE1BS0VMQU5HSUQoUFJJTUFSWUxBTkdJRChsYW5nX2lkKSwgU1VCTEFOR19ERUZBVUxUKTsKCiAgICBpZiAoIShocnNyYyA9IEZpbmRSZXNvdXJjZUV4Vygga2VybmVsMzJfaGFuZGxlLCAoTFBXU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVTG9uZ1RvUHRyKChsY3R5cGUgPj4gNCkgKyAxKSwgbGFuZ19pZCApKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfRkxBR1MgKTsgIC8qIG5vIHN1Y2ggbGN0eXBlICovCiAgICAgICAgcmV0dXJuIDA7CiAgICB9CiAgICBpZiAoIShobWVtID0gTG9hZFJlc291cmNlKCBrZXJuZWwzMl9oYW5kbGUsIGhyc3JjICkpKQogICAgICAgIHJldHVybiAwOwoKICAgIHAgPSBMb2NrUmVzb3VyY2UoIGhtZW0gKTsKICAgIGZvciAoaSA9IDA7IGkgPCAobGN0eXBlICYgMHgwZik7IGkrKykgcCArPSAqcCArIDE7CgogICAgaWYgKGxjZmxhZ3MgJiBMT0NBTEVfUkVUVVJOX05VTUJFUikgcmV0ID0gc2l6ZW9mKFVJTlQpL3NpemVvZihXQ0hBUik7CiAgICBlbHNlIHJldCA9IChsY3R5cGUgPT0gTE9DQUxFX0ZPTlRTSUdOQVRVUkUpID8gKnAgOiAqcCArIDE7CgogICAgaWYgKCFidWZmZXIpIHJldHVybiByZXQ7CgogICAgaWYgKHJldCA+IGxlbikKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBpZiAobGNmbGFncyAmIExPQ0FMRV9SRVRVUk5fTlVNQkVSKQogICAgewogICAgICAgIFVJTlQgbnVtYmVyOwogICAgICAgIFdDSEFSICplbmQsICp0bXAgPSBIZWFwQWxsb2MoIEdldFByb2Nlc3NIZWFwKCksIDAsICgqcCArIDEpICogc2l6ZW9mKFdDSEFSKSApOwogICAgICAgIGlmICghdG1wKSByZXR1cm4gMDsKICAgICAgICBtZW1jcHkoIHRtcCwgcCArIDEsICpwICogc2l6ZW9mKFdDSEFSKSApOwogICAgICAgIHRtcFsqcF0gPSAwOwogICAgICAgIG51bWJlciA9IHN0cnRvbFcoIHRtcCwgJmVuZCwgMTAgKTsKICAgICAgICBpZiAoISplbmQpCiAgICAgICAgICAgIG1lbWNweSggYnVmZmVyLCAmbnVtYmVyLCBzaXplb2YobnVtYmVyKSApOwogICAgICAgIGVsc2UgIC8qIGludmFsaWQgbnVtYmVyICovCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfRkxBR1MgKTsKICAgICAgICAgICAgcmV0ID0gMDsKICAgICAgICB9CiAgICAgICAgSGVhcEZyZWUoIEdldFByb2Nlc3NIZWFwKCksIDAsIHRtcCApOwoKICAgICAgICBUUkFDRSggIihsY2lkPTB4JXgsbGN0eXBlPTB4JXgsJXAsJWQpIHJldHVybmluZyBudW1iZXIgJWRcbiIsCiAgICAgICAgICAgICAgIGxjaWQsIGxjdHlwZSwgYnVmZmVyLCBsZW4sIG51bWJlciApOwogICAgfQogICAgZWxzZQogICAgewogICAgICAgIG1lbWNweSggYnVmZmVyLCBwICsgMSwgKnAgKiBzaXplb2YoV0NIQVIpICk7CiAgICAgICAgaWYgKGxjdHlwZSAhPSBMT0NBTEVfRk9OVFNJR05BVFVSRSkgYnVmZmVyW3JldC0xXSA9IDA7CgogICAgICAgIFRSQUNFKCAiKGxjaWQ9MHgleCxsY3R5cGU9MHgleCwlcCwlZCkgcmV0dXJuaW5nICVkICVzXG4iLAogICAgICAgICAgICAgICBsY2lkLCBsY3R5cGUsIGJ1ZmZlciwgbGVuLCByZXQsIGRlYnVnc3RyX3coYnVmZmVyKSApOwogICAgfQogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCVNldExvY2FsZUluZm9BCVtLRVJORUwzMi5AXQogKgogKiBTZXQgaW5mb3JtYXRpb24gYWJvdXQgYW4gYXNwZWN0IG9mIGEgbG9jYWxlLgogKgogKiBQQVJBTVMKICogIGxjaWQgICBbSV0gTENJRCBvZiB0aGUgbG9jYWxlCiAqICBsY3R5cGUgW0ldIExDVFlQRV8gZmxhZ3MgZnJvbSAid2lubmxzLmgiCiAqICBkYXRhICAgW0ldIEluZm9ybWF0aW9uIHRvIHNldAogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBUaGUgaW5mb3JtYXRpb24gZ2l2ZW4gd2lsbCBiZSByZXR1cm5lZCBieSBHZXRMb2NhbGVJbmZvQSgpCiAqICAgICAgICAgICB3aGVuZXZlciBpdCBpcyBjYWxsZWQgd2l0aG91dCBMT0NBTEVfTk9VU0VST1ZFUlJJREUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqCiAqIE5PVEVTCiAqICAtIFZhbHVlcyBhcmUgb25seSBiZSBzZXQgZm9yIHRoZSBjdXJyZW50IHVzZXIgbG9jYWxlOyB0aGUgc3lzdGVtIGxvY2FsZQogKiAgc2V0dGluZ3MgY2Fubm90IGJlIGNoYW5nZWQuCiAqICAtIEFueSBzZXR0aW5ncyBjaGFuZ2VkIGJ5IHRoaXMgY2FsbCBhcmUgbG9zdCB3aGVuIHRoZSBsb2NhbGUgaXMgY2hhbmdlZCBieQogKiAgdGhlIGNvbnRyb2wgcGFuZWwgKGluIFdpbmUsIHRoaXMgaGFwcGVucyBldmVyeSB0aW1lIHlvdSBjaGFuZ2UgTEFORykuCiAqICAtIFRoZSBuYXRpdmUgaW1wbGVtZW50YXRpb24gb2YgdGhpcyBmdW5jdGlvbiBkb2VzIG5vdCBjaGVjayB0aGF0IGxjaWQgbWF0Y2hlcwogKiAgdGhlIGN1cnJlbnQgdXNlciBsb2NhbGUsIGFuZCBzaW1wbHkgc2V0cyB0aGUgbmV3IHZhbHVlcy4gV2luZSB3YXJucyB5b3UgaW4KICogIHRoaXMgY2FzZSwgYnV0IGJlaGF2ZXMgdGhlIHNhbWUuCiAqLwpCT09MIFdJTkFQSSBTZXRMb2NhbGVJbmZvQShMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUsIExQQ1NUUiBkYXRhKQp7CiAgICBVSU5UIGNvZGVwYWdlID0gQ1BfQUNQOwogICAgV0NIQVIgKnN0clc7CiAgICBEV09SRCBsZW47CiAgICBCT09MIHJldDsKCiAgICBpZiAoIShsY3R5cGUgJiBMT0NBTEVfVVNFX0NQX0FDUCkpIGNvZGVwYWdlID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKCiAgICBpZiAoIWRhdGEpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KICAgIGxlbiA9IE11bHRpQnl0ZVRvV2lkZUNoYXIoIGNvZGVwYWdlLCAwLCBkYXRhLCAtMSwgTlVMTCwgMCApOwogICAgaWYgKCEoc3RyVyA9IEhlYXBBbGxvYyggR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGVuICogc2l6ZW9mKFdDSEFSKSApKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgTXVsdGlCeXRlVG9XaWRlQ2hhciggY29kZXBhZ2UsIDAsIGRhdGEsIC0xLCBzdHJXLCBsZW4gKTsKICAgIHJldCA9IFNldExvY2FsZUluZm9XKCBsY2lkLCBsY3R5cGUsIHN0clcgKTsKICAgIEhlYXBGcmVlKCBHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHJXICk7CiAgICByZXR1cm4gcmV0Owp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKgkJU2V0TG9jYWxlSW5mb1cJKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBTZXRMb2NhbGVJbmZvQS4KICovCkJPT0wgV0lOQVBJIFNldExvY2FsZUluZm9XKCBMQ0lEIGxjaWQsIExDVFlQRSBsY3R5cGUsIExQQ1dTVFIgZGF0YSApCnsKICAgIGNvbnN0IFdDSEFSICp2YWx1ZTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBpbnRsV1tdID0geydpJywnbicsJ3QnLCdsJywwIH07CiAgICBVTklDT0RFX1NUUklORyB2YWx1ZVc7CiAgICBOVFNUQVRVUyBzdGF0dXM7CiAgICBIQU5ETEUgaGtleTsKCiAgICBsY3R5cGUgJj0gMHhmZmZmOwogICAgdmFsdWUgPSBnZXRfbG9jYWxlX3ZhbHVlX25hbWUoIGxjdHlwZSApOwoKICAgIGlmICghZGF0YSB8fCAhdmFsdWUpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAobGN0eXBlID09IExPQ0FMRV9JREFURSB8fCBsY3R5cGUgPT0gTE9DQUxFX0lMREFURSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfRkxBR1MgKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgVFJBQ0UoInNldHRpbmcgJXggKCVzKSB0byAlc1xuIiwgbGN0eXBlLCBkZWJ1Z3N0cl93KHZhbHVlKSwgZGVidWdzdHJfdyhkYXRhKSApOwoKICAgIC8qIEZJWE1FOiBzaG91bGQgY2hlY2sgdGhhdCBkYXRhIHRvIHNldCBpcyBzYW5lICovCgogICAgLyogRklYTUU6IHByb2ZpbGUgZnVuY3Rpb25zIHNob3VsZCBtYXAgdG8gcmVnaXN0cnkgKi8KICAgIFdyaXRlUHJvZmlsZVN0cmluZ1coIGludGxXLCB2YWx1ZSwgZGF0YSApOwoKICAgIGlmICghKGhrZXkgPSBjcmVhdGVfcmVnaXN0cnlfa2V5KCkpKSByZXR1cm4gRkFMU0U7CiAgICBSdGxJbml0VW5pY29kZVN0cmluZyggJnZhbHVlVywgdmFsdWUgKTsKICAgIHN0YXR1cyA9IE50U2V0VmFsdWVLZXkoIGhrZXksICZ2YWx1ZVcsIDAsIFJFR19TWiwgZGF0YSwgKHN0cmxlblcoZGF0YSkrMSkqc2l6ZW9mKFdDSEFSKSApOwoKICAgIGlmIChsY3R5cGUgPT0gTE9DQUxFX1NTSE9SVERBVEUgfHwgbGN0eXBlID09IExPQ0FMRV9TTE9OR0RBVEUpCiAgICB7CiAgICAgIC8qIFNldCBJLXZhbHVlIGZyb20gUyB2YWx1ZSAqLwogICAgICBXQ0hBUiAqbHBELCAqbHBNLCAqbHBZOwogICAgICBXQ0hBUiBzekJ1ZmZbMl07CgogICAgICBscEQgPSBzdHJyY2hyVyhkYXRhLCAnZCcpOwogICAgICBscE0gPSBzdHJyY2hyVyhkYXRhLCAnTScpOwogICAgICBscFkgPSBzdHJyY2hyVyhkYXRhLCAneScpOwoKICAgICAgaWYgKGxwRCA8PSBscE0pCiAgICAgIHsKICAgICAgICBzekJ1ZmZbMF0gPSAnMSc7IC8qIEQtTS1ZICovCiAgICAgIH0KICAgICAgZWxzZQogICAgICB7CiAgICAgICAgaWYgKGxwWSA8PSBscE0pCiAgICAgICAgICBzekJ1ZmZbMF0gPSAnMic7IC8qIFktTS1EICovCiAgICAgICAgZWxzZQogICAgICAgICAgc3pCdWZmWzBdID0gJzAnOyAvKiBNLUQtWSAqLwogICAgICB9CgogICAgICBzekJ1ZmZbMV0gPSAnXDAnOwoKICAgICAgaWYgKGxjdHlwZSA9PSBMT0NBTEVfU1NIT1JUREFURSkKICAgICAgICBsY3R5cGUgPSBMT0NBTEVfSURBVEU7CiAgICAgIGVsc2UKICAgICAgICBsY3R5cGUgPSBMT0NBTEVfSUxEQVRFOwoKICAgICAgdmFsdWUgPSBnZXRfbG9jYWxlX3ZhbHVlX25hbWUoIGxjdHlwZSApOwoKICAgICAgV3JpdGVQcm9maWxlU3RyaW5nVyggaW50bFcsIHZhbHVlLCBzekJ1ZmYgKTsKCiAgICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmdmFsdWVXLCB2YWx1ZSApOwogICAgICBzdGF0dXMgPSBOdFNldFZhbHVlS2V5KCBoa2V5LCAmdmFsdWVXLCAwLCBSRUdfU1osIHN6QnVmZiwgc2l6ZW9mKHN6QnVmZikgKTsKICAgIH0KCiAgICBOdENsb3NlKCBoa2V5ICk7CgogICAgaWYgKHN0YXR1cykgU2V0TGFzdEVycm9yKCBSdGxOdFN0YXR1c1RvRG9zRXJyb3Ioc3RhdHVzKSApOwogICAgcmV0dXJuICFzdGF0dXM7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBHZXRBQ1AgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IHRoZSBjdXJyZW50IEFuc2kgY29kZSBwYWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogICAgVGhlIGN1cnJlbnQgQW5zaSBjb2RlIHBhZ2UgaWRlbnRpZmllciBmb3IgdGhlIHN5c3RlbS4KICovClVJTlQgV0lOQVBJIEdldEFDUCh2b2lkKQp7CiAgICBhc3NlcnQoIGFuc2lfY3B0YWJsZSApOwogICAgcmV0dXJuIGFuc2lfY3B0YWJsZS0+aW5mby5jb2RlcGFnZTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIFNldENQR2xvYmFsICAgKEtFUk5FTDMyLkApCiAqCiAqIFNldCB0aGUgY3VycmVudCBBbnNpIGNvZGUgcGFnZSBJZCBmb3IgdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICAgIGFjcCBbSV0gY29kZSBwYWdlIElEIHRvIGJlIHRoZSBuZXcgQUNQLgogKgogKiBSRVRVUk5TCiAqICAgIFRoZSBwcmV2aW91cyBBQ1AuCiAqLwpVSU5UIFdJTkFQSSBTZXRDUEdsb2JhbCggVUlOVCBhY3AgKQp7CiAgICBVSU5UIHJldCA9IEdldEFDUCgpOwogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqbmV3X2NwdGFibGUgPSB3aW5lX2NwX2dldF90YWJsZSggYWNwICk7CgogICAgaWYgKG5ld19jcHRhYmxlKSBhbnNpX2NwdGFibGUgPSBuZXdfY3B0YWJsZTsKICAgIHJldHVybiByZXQ7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgICAgIEdldE9FTUNQICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgY3VycmVudCBPRU0gY29kZSBwYWdlIElkIGZvciB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIE5vbmUuCiAqCiAqIFJFVFVSTlMKICogICAgVGhlIGN1cnJlbnQgT0VNIGNvZGUgcGFnZSBpZGVudGlmaWVyIGZvciB0aGUgc3lzdGVtLgogKi8KVUlOVCBXSU5BUEkgR2V0T0VNQ1Aodm9pZCkKewogICAgYXNzZXJ0KCBvZW1fY3B0YWJsZSApOwogICAgcmV0dXJuIG9lbV9jcHRhYmxlLT5pbmZvLmNvZGVwYWdlOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBJc1ZhbGlkQ29kZVBhZ2UgICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgZ2l2ZW4gY29kZSBwYWdlIGlkZW50aWZpZXIgaXMgdmFsaWQuCiAqCiAqIFBBUkFNUwogKiAgY29kZXBhZ2UgW0ldIENvZGUgcGFnZSBJZCB0byB2ZXJpZnkuCiAqCiAqIFJFVFVSTlMKICogIFRSVUUsIElmIGNvZGVwYWdlIGlzIHZhbGlkIGFuZCBhdmFpbGFibGUgb24gdGhlIHN5c3RlbSwKICogIEZBTFNFIG90aGVyd2lzZS4KICovCkJPT0wgV0lOQVBJIElzVmFsaWRDb2RlUGFnZSggVUlOVCBjb2RlcGFnZSApCnsKICAgIHN3aXRjaChjb2RlcGFnZSkgewogICAgY2FzZSBDUF9VVEY3OgogICAgY2FzZSBDUF9VVEY4OgogICAgICAgIHJldHVybiBUUlVFOwogICAgZGVmYXVsdDoKICAgICAgICByZXR1cm4gd2luZV9jcF9nZXRfdGFibGUoIGNvZGVwYWdlICkgIT0gTlVMTDsKICAgIH0KfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSXNEQkNTTGVhZEJ5dGVFeCAgIChLRVJORUwzMi5AKQogKgogKiBEZXRlcm1pbmUgaWYgYSBjaGFyYWN0ZXIgaXMgYSBsZWFkIGJ5dGUgaW4gYSBnaXZlbiBjb2RlIHBhZ2UuCiAqCiAqIFBBUkFNUwogKiAgY29kZXBhZ2UgW0ldIENvZGUgcGFnZSBmb3IgdGhlIHRlc3QuCiAqICB0ZXN0Y2hhciBbSV0gQ2hhcmFjdGVyIHRvIHRlc3QKICoKICogUkVUVVJOUwogKiAgVFJVRSwgaWYgdGVzdGNoYXIgaXMgYSBsZWFkIGJ5dGUgaW4gY29kZXBhZ2UsCiAqICBGQUxTRSBvdGhlcndpc2UuCiAqLwpCT09MIFdJTkFQSSBJc0RCQ1NMZWFkQnl0ZUV4KCBVSU5UIGNvZGVwYWdlLCBCWVRFIHRlc3RjaGFyICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGUgPSBnZXRfY29kZXBhZ2VfdGFibGUoIGNvZGVwYWdlICk7CiAgICByZXR1cm4gdGFibGUgJiYgd2luZV9pc19kYmNzX2xlYWRieXRlKCB0YWJsZSwgdGVzdGNoYXIgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSXNEQkNTTGVhZEJ5dGUgICAoS0VSTkVMMzIuQCkKICogICAgICAgICAgIElzREJDU0xlYWRCeXRlICAgKEtFUk5FTC4yMDcpCiAqCiAqIERldGVybWluZSBpZiBhIGNoYXJhY3RlciBpcyBhIGxlYWQgYnl0ZS4KICoKICogUEFSQU1TCiAqICB0ZXN0Y2hhciBbSV0gQ2hhcmFjdGVyIHRvIHRlc3QKICoKICogUkVUVVJOUwogKiAgVFJVRSwgaWYgdGVzdGNoYXIgaXMgYSBsZWFkIGJ5dGUgaW4gdGhlIEFuc2lpIGNvZGUgcGFnZSwKICogIEZBTFNFIG90aGVyd2lzZS4KICovCkJPT0wgV0lOQVBJIElzREJDU0xlYWRCeXRlKCBCWVRFIHRlc3RjaGFyICkKewogICAgaWYgKCFhbnNpX2NwdGFibGUpIHJldHVybiBGQUxTRTsKICAgIHJldHVybiB3aW5lX2lzX2RiY3NfbGVhZGJ5dGUoIGFuc2lfY3B0YWJsZSwgdGVzdGNoYXIgKTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0Q1BJbmZvICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBpbmZvcm1hdGlvbiBhYm91dCBhIGNvZGUgcGFnZS4KICoKICogUEFSQU1TCiAqICBjb2RlcGFnZSBbSV0gQ29kZSBwYWdlIG51bWJlcgogKiAgY3BpbmZvICAgW09dIERlc3RpbmF0aW9uIGZvciBjb2RlIHBhZ2UgaW5mb3JtYXRpb24KICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4gY3BpbmZvIGlzIHVwZGF0ZWQgd2l0aCB0aGUgaW5mb3JtYXRpb24gYWJvdXQgY29kZXBhZ2UuCiAqICBGYWlsdXJlOiBGQUxTRSwgaWYgY29kZXBhZ2UgaXMgaW52YWxpZCBvciBjcGluZm8gaXMgTlVMTC4KICovCkJPT0wgV0lOQVBJIEdldENQSW5mbyggVUlOVCBjb2RlcGFnZSwgTFBDUElORk8gY3BpbmZvICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CgogICAgaWYgKCFjcGluZm8pCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAoISh0YWJsZSA9IGdldF9jb2RlcGFnZV90YWJsZSggY29kZXBhZ2UgKSkpCiAgICB7CiAgICAgICAgc3dpdGNoKGNvZGVwYWdlKQogICAgICAgIHsKICAgICAgICAgICAgY2FzZSBDUF9VVEY3OgogICAgICAgICAgICBjYXNlIENQX1VURjg6CiAgICAgICAgICAgICAgICBjcGluZm8tPkRlZmF1bHRDaGFyWzBdID0gMHgzZjsKICAgICAgICAgICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMV0gPSAwOwogICAgICAgICAgICAgICAgY3BpbmZvLT5MZWFkQnl0ZVswXSA9IGNwaW5mby0+TGVhZEJ5dGVbMV0gPSAwOwogICAgICAgICAgICAgICAgY3BpbmZvLT5NYXhDaGFyU2l6ZSA9IChjb2RlcGFnZSA9PSBDUF9VVEY3KSA/IDUgOiA0OwogICAgICAgICAgICAgICAgcmV0dXJuIFRSVUU7CiAgICAgICAgfQoKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQogICAgaWYgKHRhYmxlLT5pbmZvLmRlZl9jaGFyICYgMHhmZjAwKQogICAgewogICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMF0gPSAodGFibGUtPmluZm8uZGVmX2NoYXIgJiAweGZmMDApID4+IDg7CiAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclsxXSA9IHRhYmxlLT5pbmZvLmRlZl9jaGFyICYgMHgwMGZmOwogICAgfQogICAgZWxzZQogICAgewogICAgICAgIGNwaW5mby0+RGVmYXVsdENoYXJbMF0gPSB0YWJsZS0+aW5mby5kZWZfY2hhciAmIDB4ZmY7CiAgICAgICAgY3BpbmZvLT5EZWZhdWx0Q2hhclsxXSA9IDA7CiAgICB9CiAgICBpZiAoKGNwaW5mby0+TWF4Q2hhclNpemUgPSB0YWJsZS0+aW5mby5jaGFyX3NpemUpID09IDIpCiAgICAgICAgbWVtY3B5KCBjcGluZm8tPkxlYWRCeXRlLCB0YWJsZS0+ZGJjcy5sZWFkX2J5dGVzLCBzaXplb2YoY3BpbmZvLT5MZWFkQnl0ZSkgKTsKICAgIGVsc2UKICAgICAgICBjcGluZm8tPkxlYWRCeXRlWzBdID0gY3BpbmZvLT5MZWFkQnl0ZVsxXSA9IDA7CgogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0Q1BJbmZvRXhBICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBleHRlbmRlZCBpbmZvcm1hdGlvbiBhYm91dCBhIGNvZGUgcGFnZS4KICoKICogUEFSQU1TCiAqICBjb2RlcGFnZSBbSV0gQ29kZSBwYWdlIG51bWJlcgogKiAgZHdGbGFncyAgW0ldIFJlc2VydmVkLCBtdXN0IHRvIDAuCiAqICBjcGluZm8gICBbT10gRGVzdGluYXRpb24gZm9yIGNvZGUgcGFnZSBpbmZvcm1hdGlvbgogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBjcGluZm8gaXMgdXBkYXRlZCB3aXRoIHRoZSBpbmZvcm1hdGlvbiBhYm91dCBjb2RlcGFnZS4KICogIEZhaWx1cmU6IEZBTFNFLCBpZiBjb2RlcGFnZSBpcyBpbnZhbGlkIG9yIGNwaW5mbyBpcyBOVUxMLgogKi8KQk9PTCBXSU5BUEkgR2V0Q1BJbmZvRXhBKCBVSU5UIGNvZGVwYWdlLCBEV09SRCBkd0ZsYWdzLCBMUENQSU5GT0VYQSBjcGluZm8gKQp7CiAgICBDUElORk9FWFcgY3BpbmZvVzsKCiAgICBpZiAoIUdldENQSW5mb0V4VyggY29kZXBhZ2UsIGR3RmxhZ3MsICZjcGluZm9XICkpCiAgICAgIHJldHVybiBGQUxTRTsKCiAgICAvKiB0aGUgbGF5b3V0IGlzIHRoZSBzYW1lIGV4Y2VwdCBmb3IgQ29kZVBhZ2VOYW1lICovCiAgICBtZW1jcHkoY3BpbmZvLCAmY3BpbmZvVywgc2l6ZW9mKENQSU5GT0VYQSkpOwogICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIGNwaW5mb1cuQ29kZVBhZ2VOYW1lLCAtMSwgY3BpbmZvLT5Db2RlUGFnZU5hbWUsIHNpemVvZihjcGluZm8tPkNvZGVQYWdlTmFtZSksIE5VTEwsIE5VTEwpOwogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0Q1BJbmZvRXhXICAgKEtFUk5FTDMyLkApCiAqCiAqIFVuaWNvZGUgdmVyc2lvbiBvZiBHZXRDUEluZm9FeEEuCiAqLwpCT09MIFdJTkFQSSBHZXRDUEluZm9FeFcoIFVJTlQgY29kZXBhZ2UsIERXT1JEIGR3RmxhZ3MsIExQQ1BJTkZPRVhXIGNwaW5mbyApCnsKICAgIGlmICghR2V0Q1BJbmZvKCBjb2RlcGFnZSwgKExQQ1BJTkZPKWNwaW5mbyApKQogICAgICByZXR1cm4gRkFMU0U7CgogICAgc3dpdGNoKGNvZGVwYWdlKQogICAgewogICAgICAgIGNhc2UgQ1BfVVRGNzoKICAgICAgICB7CiAgICAgICAgICAgIHN0YXRpYyBjb25zdCBXQ0hBUiB1dGY3W10gPSB7J1UnLCduJywnaScsJ2MnLCdvJywnZCcsJ2UnLCcgJywnKCcsJ1UnLCdUJywnRicsJy0nLCc3JywnKScsMH07CgogICAgICAgICAgICBjcGluZm8tPkNvZGVQYWdlID0gQ1BfVVRGNzsKICAgICAgICAgICAgY3BpbmZvLT5Vbmljb2RlRGVmYXVsdENoYXIgPSAweDNmOwogICAgICAgICAgICBzdHJjcHlXKGNwaW5mby0+Q29kZVBhZ2VOYW1lLCB1dGY3KTsKICAgICAgICAgICAgYnJlYWs7CiAgICAgICAgfQoKICAgICAgICBjYXNlIENQX1VURjg6CiAgICAgICAgewogICAgICAgICAgICBzdGF0aWMgY29uc3QgV0NIQVIgdXRmOFtdID0geydVJywnbicsJ2knLCdjJywnbycsJ2QnLCdlJywnICcsJygnLCdVJywnVCcsJ0YnLCctJywnOCcsJyknLDB9OwoKICAgICAgICAgICAgY3BpbmZvLT5Db2RlUGFnZSA9IENQX1VURjg7CiAgICAgICAgICAgIGNwaW5mby0+VW5pY29kZURlZmF1bHRDaGFyID0gMHgzZjsKICAgICAgICAgICAgc3RyY3B5VyhjcGluZm8tPkNvZGVQYWdlTmFtZSwgdXRmOCk7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KCiAgICAgICAgZGVmYXVsdDoKICAgICAgICB7CiAgICAgICAgICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBjb2RlcGFnZSApOwoKICAgICAgICAgICAgY3BpbmZvLT5Db2RlUGFnZSA9IHRhYmxlLT5pbmZvLmNvZGVwYWdlOwogICAgICAgICAgICBjcGluZm8tPlVuaWNvZGVEZWZhdWx0Q2hhciA9IHRhYmxlLT5pbmZvLmRlZl91bmljb2RlX2NoYXI7CiAgICAgICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIoIENQX0FDUCwgMCwgdGFibGUtPmluZm8ubmFtZSwgLTEsIGNwaW5mby0+Q29kZVBhZ2VOYW1lLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzaXplb2YoY3BpbmZvLT5Db2RlUGFnZU5hbWUpL3NpemVvZihXQ0hBUikpOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICB9CiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBFbnVtU3lzdGVtQ29kZVBhZ2VzQSAgIChLRVJORUwzMi5AKQogKgogKiBDYWxsIGEgdXNlciBkZWZpbmVkIGZ1bmN0aW9uIGZvciBldmVyeSBjb2RlIHBhZ2UgaW5zdGFsbGVkIG9uIHRoZSBzeXN0ZW0uCiAqCiAqIFBBUkFNUwogKiAgIGxwZm5Db2RlUGFnZUVudW0gW0ldIFVzZXIgQ09ERVBBR0VfRU5VTVBST0MgdG8gY2FsbCB3aXRoIGVhY2ggZm91bmQgY29kZSBwYWdlCiAqICAgZmxhZ3MgICAgICAgICAgICBbSV0gUmVzZXJ2ZWQsIHNldCB0byAwLgogKgogKiBSRVRVUk5TCiAqICBUUlVFLCBJZiBhbGwgY29kZSBwYWdlcyBoYXZlIGJlZW4gZW51bWVyYXRlZCwgb3IKICogIEZBTFNFIGlmIGxwZm5Db2RlUGFnZUVudW0gcmV0dXJuZWQgRkFMU0UgdG8gc3RvcCB0aGUgZW51bWVyYXRpb24uCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtQ29kZVBhZ2VzQSggQ09ERVBBR0VfRU5VTVBST0NBIGxwZm5Db2RlUGFnZUVudW0sIERXT1JEIGZsYWdzICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CiAgICBjaGFyIGJ1ZmZlclsxMF07CiAgICBpbnQgaW5kZXggPSAwOwoKICAgIGZvciAoOzspCiAgICB7CiAgICAgICAgaWYgKCEodGFibGUgPSB3aW5lX2NwX2VudW1fdGFibGUoIGluZGV4KysgKSkpIGJyZWFrOwogICAgICAgIHNwcmludGYoIGJ1ZmZlciwgIiVkIiwgdGFibGUtPmluZm8uY29kZXBhZ2UgKTsKICAgICAgICBpZiAoIWxwZm5Db2RlUGFnZUVudW0oIGJ1ZmZlciApKSBicmVhazsKICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBFbnVtU3lzdGVtQ29kZVBhZ2VzVyAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRW51bVN5c3RlbUNvZGVQYWdlc0EuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtQ29kZVBhZ2VzVyggQ09ERVBBR0VfRU5VTVBST0NXIGxwZm5Db2RlUGFnZUVudW0sIERXT1JEIGZsYWdzICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CiAgICBXQ0hBUiBidWZmZXJbMTBdLCAqcDsKICAgIGludCBwYWdlLCBpbmRleCA9IDA7CgogICAgZm9yICg7OykKICAgIHsKICAgICAgICBpZiAoISh0YWJsZSA9IHdpbmVfY3BfZW51bV90YWJsZSggaW5kZXgrKyApKSkgYnJlYWs7CiAgICAgICAgcCA9IGJ1ZmZlciArIHNpemVvZihidWZmZXIpL3NpemVvZihXQ0hBUik7CiAgICAgICAgKi0tcCA9IDA7CiAgICAgICAgcGFnZSA9IHRhYmxlLT5pbmZvLmNvZGVwYWdlOwogICAgICAgIGRvCiAgICAgICAgewogICAgICAgICAgICAqLS1wID0gJzAnICsgKHBhZ2UgJSAxMCk7CiAgICAgICAgICAgIHBhZ2UgLz0gMTA7CiAgICAgICAgfSB3aGlsZSggcGFnZSApOwogICAgICAgIGlmICghbHBmbkNvZGVQYWdlRW51bSggcCApKSBicmVhazsKICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICAgICBNdWx0aUJ5dGVUb1dpZGVDaGFyICAgKEtFUk5FTDMyLkApCiAqCiAqIENvbnZlcnQgYSBtdWx0aWJ5dGUgY2hhcmFjdGVyIHN0cmluZyBpbnRvIGEgVW5pY29kZSBzdHJpbmcuCiAqCiAqIFBBUkFNUwogKiAgIHBhZ2UgICBbSV0gQ29kZXBhZ2UgY2hhcmFjdGVyIHNldCB0byBjb252ZXJ0IGZyb20KICogICBmbGFncyAgW0ldIENoYXJhY3RlciBtYXBwaW5nIGZsYWdzCiAqICAgc3JjICAgIFtJXSBTb3VyY2Ugc3RyaW5nIGJ1ZmZlcgogKiAgIHNyY2xlbiBbSV0gTGVuZ3RoIG9mIHNyYyAoaW4gYnl0ZXMpLCBvciAtMSBpZiBzcmMgaXMgTlVMIHRlcm1pbmF0ZWQKICogICBkc3QgICAgW09dIERlc3RpbmF0aW9uIGJ1ZmZlcgogKiAgIGRzdGxlbiBbSV0gTGVuZ3RoIG9mIGRzdCAoaW4gV0NIQVJzKSwgb3IgMCB0byBjb21wdXRlIHRoZSByZXF1aXJlZCBsZW5ndGgKICoKICogUkVUVVJOUwogKiAgIFN1Y2Nlc3M6IElmIGRzdGxlbiA+IDAsIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyB3cml0dGVuIHRvIGRzdC4KICogICAgICAgICAgICBJZiBkc3RsZW4gPT0gMCwgdGhlIG51bWJlciBvZiBjaGFyYWN0ZXJzIG5lZWRlZCB0byBwZXJmb3JtIHRoZQogKiAgICAgICAgICAgIGNvbnZlcnNpb24uIEluIGJvdGggY2FzZXMgdGhlIGNvdW50IGluY2x1ZGVzIHRoZSB0ZXJtaW5hdGluZyBOVUwuCiAqICAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuIFBvc3NpYmxlIGVycm9ycyBhcmUKICogICAgICAgICAgICBFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSLCBpZiBub3QgZW5vdWdoIHNwYWNlIGlzIGF2YWlsYWJsZSBpbiBkc3QKICogICAgICAgICAgICBhbmQgZHN0bGVuICE9IDA7IEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSLCAgaWYgYW4gaW52YWxpZCBwYXJhbWV0ZXIKICogICAgICAgICAgICBpcyBwYXNzZWQsIGFuZCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OIGlmIG5vIHRyYW5zbGF0aW9uIGlzCiAqICAgICAgICAgICAgcG9zc2libGUgZm9yIHNyYy4KICovCklOVCBXSU5BUEkgTXVsdGlCeXRlVG9XaWRlQ2hhciggVUlOVCBwYWdlLCBEV09SRCBmbGFncywgTFBDU1RSIHNyYywgSU5UIHNyY2xlbiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUFdTVFIgZHN0LCBJTlQgZHN0bGVuICkKewogICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdGFibGU7CiAgICBpbnQgcmV0OwoKICAgIGlmICghc3JjIHx8ICghZHN0ICYmIGRzdGxlbikpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmIChzcmNsZW4gPCAwKSBzcmNsZW4gPSBzdHJsZW4oc3JjKSArIDE7CgogICAgc3dpdGNoKHBhZ2UpCiAgICB7CiAgICBjYXNlIENQX1NZTUJPTDoKICAgICAgICBpZiggZmxhZ3MpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KICAgICAgICByZXQgPSB3aW5lX2Nwc3ltYm9sX21ic3Rvd2NzKCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4gKTsKICAgICAgICBicmVhazsKICAgIGNhc2UgQ1BfVVRGNzoKICAgICAgICBGSVhNRSgiVVRGLTcgbm90IHN1cHBvcnRlZFxuIik7CiAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9DQUxMX05PVF9JTVBMRU1FTlRFRCApOwogICAgICAgIHJldHVybiAwOwogICAgY2FzZSBDUF9VTklYQ1A6CiAgICAgICAgaWYgKHVuaXhfY3B0YWJsZSkKICAgICAgICB7CiAgICAgICAgICAgIHJldCA9IHdpbmVfY3BfbWJzdG93Y3MoIHVuaXhfY3B0YWJsZSwgZmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbiApOwogICAgICAgICAgICBicmVhazsKICAgICAgICB9CiAgICAgICAgLyogZmFsbCB0aHJvdWdoICovCiAgICBjYXNlIENQX1VURjg6CiAgICAgICAgcmV0ID0gd2luZV91dGY4X21ic3Rvd2NzKCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIGlmICghKHRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBwYWdlICkpKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9jcF9tYnN0b3djcyggdGFibGUsIGZsYWdzLCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4gKTsKICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAocmV0IDwgMCkKICAgIHsKICAgICAgICBzd2l0Y2gocmV0KQogICAgICAgIHsKICAgICAgICBjYXNlIC0xOiBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsgYnJlYWs7CiAgICAgICAgY2FzZSAtMjogU2V0TGFzdEVycm9yKCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OICk7IGJyZWFrOwogICAgICAgIH0KICAgICAgICByZXQgPSAwOwogICAgfQogICAgVFJBQ0UoImNwICVkICVzIC0+ICVzLCByZXQgPSAlZFxuIiwKICAgICAgICAgIHBhZ2UsIGRlYnVnc3RyX2FuKHNyYywgc3JjbGVuKSwgZGVidWdzdHJfd24oZHN0LCByZXQpLCByZXQpOwogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZSAgIChLRVJORUwzMi5AKQogKgogKiBDb252ZXJ0IGEgVW5pY29kZSBjaGFyYWN0ZXIgc3RyaW5nIGludG8gYSBtdWx0aWJ5dGUgc3RyaW5nLgogKgogKiBQQVJBTVMKICogICBwYWdlICAgIFtJXSBDb2RlIHBhZ2UgY2hhcmFjdGVyIHNldCB0byBjb252ZXJ0IHRvCiAqICAgZmxhZ3MgICBbSV0gTWFwcGluZyBGbGFncyAoTUJfIGNvbnN0YW50cyBmcm9tICJ3aW5ubHMuaCIpLgogKiAgIHNyYyAgICAgW0ldIFNvdXJjZSBzdHJpbmcgYnVmZmVyCiAqICAgc3JjbGVuICBbSV0gTGVuZ3RoIG9mIHNyYyAoaW4gV0NIQVJzKSwgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICAgZHN0ICAgICBbT10gRGVzdGluYXRpb24gYnVmZmVyCiAqICAgZHN0bGVuICBbSV0gTGVuZ3RoIG9mIGRzdCAoaW4gYnl0ZXMpLCBvciAwIHRvIGNvbXB1dGUgdGhlIHJlcXVpcmVkIGxlbmd0aAogKiAgIGRlZmNoYXIgW0ldIERlZmF1bHQgY2hhcmFjdGVyIHRvIHVzZSBmb3IgY29udmVyc2lvbiBpZiBubyBleGFjdAogKgkJICAgIGNvbnZlcnNpb24gY2FuIGJlIG1hZGUKICogICB1c2VkICAgIFtPXSBTZXQgaWYgZGVmYXVsdCBjaGFyYWN0ZXIgd2FzIHVzZWQgaW4gdGhlIGNvbnZlcnNpb24KICoKICogUkVUVVJOUwogKiAgIFN1Y2Nlc3M6IElmIGRzdGxlbiA+IDAsIHRoZSBudW1iZXIgb2YgY2hhcmFjdGVycyB3cml0dGVuIHRvIGRzdC4KICogICAgICAgICAgICBJZiBkc3RsZW4gPT0gMCwgbnVtYmVyIG9mIGNoYXJhY3RlcnMgbmVlZGVkIHRvIHBlcmZvcm0gdGhlCiAqICAgICAgICAgICAgY29udmVyc2lvbi4gSW4gYm90aCBjYXNlcyB0aGUgY291bnQgaW5jbHVkZXMgdGhlIHRlcm1pbmF0aW5nIE5VTC4KICogICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4gUG9zc2libGUgZXJyb3JzIGFyZQogKiAgICAgICAgICAgIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIsIGlmIG5vdCBlbm91Z2ggc3BhY2UgaXMgYXZhaWxhYmxlIGluIGRzdAogKiAgICAgICAgICAgIGFuZCBkc3RsZW4gIT0gMCwgYW5kIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSLCBpZiBhbiBpbnZhbGlkCiAqICAgICAgICAgICAgcGFyYW1ldGVyIHdhcyBnaXZlbi4KICovCklOVCBXSU5BUEkgV2lkZUNoYXJUb011bHRpQnl0ZSggVUlOVCBwYWdlLCBEV09SRCBmbGFncywgTFBDV1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBTVFIgZHN0LCBJTlQgZHN0bGVuLCBMUENTVFIgZGVmY2hhciwgQk9PTCAqdXNlZCApCnsKICAgIGNvbnN0IHVuaW9uIGNwdGFibGUgKnRhYmxlOwogICAgaW50IHJldCwgdXNlZF90bXA7CgogICAgaWYgKCFzcmMgfHwgKCFkc3QgJiYgZHN0bGVuKSkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSICk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKHNyY2xlbiA8IDApIHNyY2xlbiA9IHN0cmxlblcoc3JjKSArIDE7CgogICAgc3dpdGNoKHBhZ2UpCiAgICB7CiAgICBjYXNlIENQX1NZTUJPTDoKICAgICAgICBpZiggZmxhZ3MgfHwgZGVmY2hhciB8fCB1c2VkKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9jcHN5bWJvbF93Y3N0b21icyggc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICBjYXNlIENQX1VURjc6CiAgICAgICAgRklYTUUoIlVURi03IG5vdCBzdXBwb3J0ZWRcbiIpOwogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfQ0FMTF9OT1RfSU1QTEVNRU5URUQgKTsKICAgICAgICByZXR1cm4gMDsKICAgIGNhc2UgQ1BfVU5JWENQOgogICAgICAgIGlmICh1bml4X2NwdGFibGUpCiAgICAgICAgewogICAgICAgICAgICByZXQgPSB3aW5lX2NwX3djc3RvbWJzKCB1bml4X2NwdGFibGUsIGZsYWdzLCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRlZmNoYXIsIHVzZWQgPyAmdXNlZF90bXAgOiBOVUxMICk7CiAgICAgICAgICAgIGJyZWFrOwogICAgICAgIH0KICAgICAgICAvKiBmYWxsIHRocm91Z2ggKi8KICAgIGNhc2UgQ1BfVVRGODoKICAgICAgICBpZiAodXNlZCkgKnVzZWQgPSBGQUxTRTsgIC8qIGFsbCBjaGFycyBhcmUgdmFsaWQgZm9yIFVURi04ICovCiAgICAgICAgcmV0ID0gd2luZV91dGY4X3djc3RvbWJzKCBmbGFncywgc3JjLCBzcmNsZW4sIGRzdCwgZHN0bGVuICk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIGlmICghKHRhYmxlID0gZ2V0X2NvZGVwYWdlX3RhYmxlKCBwYWdlICkpKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKCBFUlJPUl9JTlZBTElEX1BBUkFNRVRFUiApOwogICAgICAgICAgICByZXR1cm4gMDsKICAgICAgICB9CiAgICAgICAgcmV0ID0gd2luZV9jcF93Y3N0b21icyggdGFibGUsIGZsYWdzLCBzcmMsIHNyY2xlbiwgZHN0LCBkc3RsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVmY2hhciwgdXNlZCA/ICZ1c2VkX3RtcCA6IE5VTEwgKTsKICAgICAgICBpZiAodXNlZCkgKnVzZWQgPSB1c2VkX3RtcDsKICAgICAgICBicmVhazsKICAgIH0KCiAgICBpZiAocmV0IDwgMCkKICAgIHsKICAgICAgICBzd2l0Y2gocmV0KQogICAgICAgIHsKICAgICAgICBjYXNlIC0xOiBTZXRMYXN0RXJyb3IoIEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIgKTsgYnJlYWs7CiAgICAgICAgY2FzZSAtMjogU2V0TGFzdEVycm9yKCBFUlJPUl9OT19VTklDT0RFX1RSQU5TTEFUSU9OICk7IGJyZWFrOwogICAgICAgIH0KICAgICAgICByZXQgPSAwOwogICAgfQogICAgVFJBQ0UoImNwICVkICVzIC0+ICVzLCByZXQgPSAlZFxuIiwKICAgICAgICAgIHBhZ2UsIGRlYnVnc3RyX3duKHNyYywgc3JjbGVuKSwgZGVidWdzdHJfYW4oZHN0LCByZXQpLCByZXQpOwogICAgcmV0dXJuIHJldDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgR2V0VGhyZWFkTG9jYWxlICAgIChLRVJORUwzMi5AKQogKgogKiBHZXQgdGhlIGN1cnJlbnQgdGhyZWFkcyBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgVGhlIExDSUQgY3VycmVudGx5IGFzc29jYXRlZCB3aXRoIHRoZSBjYWxsaW5nIHRocmVhZC4KICovCkxDSUQgV0lOQVBJIEdldFRocmVhZExvY2FsZSh2b2lkKQp7CiAgICBMQ0lEIHJldCA9IE50Q3VycmVudFRlYigpLT5DdXJyZW50TG9jYWxlOwogICAgaWYgKCFyZXQpIE50Q3VycmVudFRlYigpLT5DdXJyZW50TG9jYWxlID0gcmV0ID0gR2V0VXNlckRlZmF1bHRMQ0lEKCk7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgU2V0VGhyZWFkTG9jYWxlICAgIChLRVJORUwzMi5AKQogKgogKiBTZXQgdGhlIGN1cnJlbnQgdGhyZWFkcyBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgbGNpZCBbSV0gTENJRCBvZiB0aGUgbG9jYWxlIHRvIHNldAogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBUaGUgdGhyZWFkcyBsb2NhbGUgaXMgc2V0IHRvIGxjaWQuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBTZXRUaHJlYWRMb2NhbGUoIExDSUQgbGNpZCApCnsKICAgIFRSQUNFKCIoMHglMDRYKVxuIiwgbGNpZCk7CgogICAgbGNpZCA9IENvbnZlcnREZWZhdWx0TG9jYWxlKGxjaWQpOwoKICAgIGlmIChsY2lkICE9IEdldFRocmVhZExvY2FsZSgpKQogICAgewogICAgICAgIGlmICghSXNWYWxpZExvY2FsZShsY2lkLCBMQ0lEX1NVUFBPUlRFRCkpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgICAgICByZXR1cm4gRkFMU0U7CiAgICAgICAgfQoKICAgICAgICBOdEN1cnJlbnRUZWIoKS0+Q3VycmVudExvY2FsZSA9IGxjaWQ7CiAgICAgICAga2VybmVsX2dldF90aHJlYWRfZGF0YSgpLT5jb2RlX3BhZ2UgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwogICAgfQogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBTZXRUaHJlYWRVSUxhbmd1YWdlICAgIChLRVJORUwzMi5AKQogKgogKiBTZXQgdGhlIGN1cnJlbnQgdGhyZWFkcyBVSSBsYW5ndWFnZS4KICoKICogUEFSQU1TCiAqICBsYW5naWQgW0ldIExBTkdJRCBvZiB0aGUgbGFuZ3VhZ2UgdG8gc2V0LCBvciAwIHRvIHVzZQogKiAgICAgICAgICAgICB0aGUgYXZhaWxhYmxlIGxhbmd1YWdlIHdoaWNoIGlzIGJlc3Qgc3VwcG9ydGVkCiAqICAgICAgICAgICAgIGZvciBjb25zb2xlIGFwcGxpY2F0aW9ucwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgcmV0dXJuIHZhbHVlIGlzIHRoZSBzYW1lIGFzIHRoZSBpbnB1dCB2YWx1ZS4KICogIEZhaWx1cmU6IFRoZSByZXR1cm4gdmFsdWUgZGlmZmVycyBmcm9tIHRoZSBpbnB1dCB2YWx1ZS4KICogICAgICAgICAgIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KTEFOR0lEIFdJTkFQSSBTZXRUaHJlYWRVSUxhbmd1YWdlKCBMQU5HSUQgbGFuZ2lkICkKewogICAgVFJBQ0UoIigweCUwNHgpIHN0dWIgLSByZXR1cm5pbmcgc3VjY2Vzc1xuIiwgbGFuZ2lkKTsKICAgIHJldHVybiBsYW5naWQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICoJCUNvbnZlcnREZWZhdWx0TG9jYWxlIChLRVJORUwzMi5AKQogKgogKiBDb252ZXJ0IGEgZGVmYXVsdCBsb2NhbGUgaWRlbnRpZmllciBpbnRvIGEgcmVhbCBpZGVudGlmaWVyLgogKgogKiBQQVJBTVMKICogIGxjaWQgW0ldIExDSUQgaWRlbnRpZmllciBvZiB0aGUgbG9jYWxlIHRvIGNvbnZlcnQKICoKICogUkVUVVJOUwogKiAgbGNpZCB1bmNoYW5nZWQsIGlmIG5vdCBhIGRlZmF1bHQgbG9jYWxlIG9yIGl0cyBzdWJsYW5ndWFnZSBpcwogKiAgIG5vdCBTVUJMQU5HX05FVVRSQUwuCiAqICBHZXRTeXN0ZW1EZWZhdWx0TENJRCgpLCBpZiBsY2lkID09IExPQ0FMRV9TWVNURU1fREVGQVVMVC4KICogIEdldFVzZXJEZWZhdWx0TENJRCgpLCBpZiBsY2lkID09IExPQ0FMRV9VU0VSX0RFRkFVTFQgb3IgTE9DQUxFX05FVVRSQUwuCiAqICBPdGhlcndpc2UsIGxjaWQgd2l0aCBzdWJsYW5ndWFnZSBjaGFuZ2VkIHRvIFNVQkxBTkdfREVGQVVMVC4KICovCkxDSUQgV0lOQVBJIENvbnZlcnREZWZhdWx0TG9jYWxlKCBMQ0lEIGxjaWQgKQp7CiAgICBMQU5HSUQgbGFuZ2lkOwoKICAgIHN3aXRjaCAobGNpZCkKICAgIHsKICAgIGNhc2UgTE9DQUxFX1NZU1RFTV9ERUZBVUxUOgogICAgICAgIGxjaWQgPSBHZXRTeXN0ZW1EZWZhdWx0TENJRCgpOwogICAgICAgIGJyZWFrOwogICAgY2FzZSBMT0NBTEVfVVNFUl9ERUZBVUxUOgogICAgY2FzZSBMT0NBTEVfTkVVVFJBTDoKICAgICAgICBsY2lkID0gR2V0VXNlckRlZmF1bHRMQ0lEKCk7CiAgICAgICAgYnJlYWs7CiAgICBkZWZhdWx0OgogICAgICAgIC8qIFJlcGxhY2UgU1VCTEFOR19ORVVUUkFMIHdpdGggU1VCTEFOR19ERUZBVUxUICovCiAgICAgICAgbGFuZ2lkID0gTEFOR0lERlJPTUxDSUQobGNpZCk7CiAgICAgICAgaWYgKFNVQkxBTkdJRChsYW5naWQpID09IFNVQkxBTkdfTkVVVFJBTCkKICAgICAgICB7CiAgICAgICAgICBsYW5naWQgPSBNQUtFTEFOR0lEKFBSSU1BUllMQU5HSUQobGFuZ2lkKSwgU1VCTEFOR19ERUZBVUxUKTsKICAgICAgICAgIGxjaWQgPSBNQUtFTENJRChsYW5naWQsIFNPUlRJREZST01MQ0lEKGxjaWQpKTsKICAgICAgICB9CiAgICB9CiAgICByZXR1cm4gbGNpZDsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIElzVmFsaWRMb2NhbGUgICAoS0VSTkVMMzIuQCkKICoKICogRGV0ZXJtaW5lIGlmIGEgbG9jYWxlIGlzIHZhbGlkLgogKgogKiBQQVJBTVMKICogIGxjaWQgIFtJXSBMQ0lEIG9mIHRoZSBsb2NhbGUgdG8gY2hlY2sKICogIGZsYWdzIFtJXSBMQ0lEX1NVUFBPUlRFRCA9IFZhbGlkLCBMQ0lEX0lOU1RBTExFRCA9IFZhbGlkIGFuZCBpbnN0YWxsZWQgb24gdGhlIHN5c3RlbQogKgogKiBSRVRVUk5TCiAqICBUUlVFLCAgaWYgbGNpZCBpcyB2YWxpZCwKICogIEZBTFNFLCBvdGhlcndpc2UuCiAqCiAqIE5PVEVTCiAqICBXaW5lIGRvZXMgbm90IGN1cnJlbnRseSBtYWtlIHRoZSBkaXN0aW5jdGlvbiBiZXR3ZWVuIHN1cHBvcnRlZCBhbmQgaW5zdGFsbGVkLiBBbGwKICogIGxhbmd1YWdlcyBzdXBwb3J0ZWQgYXJlIGluc3RhbGxlZCBieSBkZWZhdWx0LgogKi8KQk9PTCBXSU5BUEkgSXNWYWxpZExvY2FsZSggTENJRCBsY2lkLCBEV09SRCBmbGFncyApCnsKICAgIC8qIGNoZWNrIGlmIGxhbmd1YWdlIGlzIHJlZ2lzdGVyZWQgaW4gdGhlIGtlcm5lbDMyIHJlc291cmNlcyAqLwogICAgcmV0dXJuIEZpbmRSZXNvdXJjZUV4Vygga2VybmVsMzJfaGFuZGxlLCAoTFBXU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENXU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIExBTkdJREZST01MQ0lEKGxjaWQpKSAhPSAwOwp9CgoKc3RhdGljIEJPT0wgQ0FMTEJBQ0sgZW51bV9sYW5nX3Byb2NfYSggSE1PRFVMRSBoTW9kdWxlLCBMUENTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDU1RSIG5hbWUsIFdPUkQgTGFuZ0lELCBMT05HX1BUUiBsUGFyYW0gKQp7CiAgICBMT0NBTEVfRU5VTVBST0NBIGxwZm5Mb2NhbGVFbnVtID0gKExPQ0FMRV9FTlVNUFJPQ0EpbFBhcmFtOwogICAgY2hhciBidWZbMjBdOwoKICAgIHNwcmludGYoYnVmLCAiJTA4eCIsIChVSU5UKUxhbmdJRCk7CiAgICByZXR1cm4gbHBmbkxvY2FsZUVudW0oIGJ1ZiApOwp9CgpzdGF0aWMgQk9PTCBDQUxMQkFDSyBlbnVtX2xhbmdfcHJvY193KCBITU9EVUxFIGhNb2R1bGUsIExQQ1dTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBDV1NUUiBuYW1lLCBXT1JEIExhbmdJRCwgTE9OR19QVFIgbFBhcmFtICkKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIGZvcm1hdFdbXSA9IHsnJScsJzAnLCc4JywneCcsMH07CiAgICBMT0NBTEVfRU5VTVBST0NXIGxwZm5Mb2NhbGVFbnVtID0gKExPQ0FMRV9FTlVNUFJPQ1cpbFBhcmFtOwogICAgV0NIQVIgYnVmWzIwXTsKICAgIHNwcmludGZXKCBidWYsIGZvcm1hdFcsIChVSU5UKUxhbmdJRCApOwogICAgcmV0dXJuIGxwZm5Mb2NhbGVFbnVtKCBidWYgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxvY2FsZXNBICAoS0VSTkVMMzIuQCkKICoKICogQ2FsbCBhIHVzZXJzIGZ1bmN0aW9uIGZvciBlYWNoIGxvY2FsZSBhdmFpbGFibGUgb24gdGhlIHN5c3RlbS4KICoKICogUEFSQU1TCiAqICBscGZuTG9jYWxlRW51bSBbSV0gQ2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCBmb3IgZWFjaCBsb2NhbGUKICogIGR3RmxhZ3MgICAgICAgIFtJXSBMT0NBTEVfU1VQUE9SVEVEPUFsbCBzdXBwb3J0ZWQsIExPQ0FMRV9JTlNUQUxMRUQ9SW5zdGFsbGVkIG9ubHkKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEVudW1TeXN0ZW1Mb2NhbGVzQSggTE9DQUxFX0VOVU1QUk9DQSBscGZuTG9jYWxlRW51bSwgRFdPUkQgZHdGbGFncyApCnsKICAgIFRSQUNFKCIoJXAsJTA4eClcbiIsIGxwZm5Mb2NhbGVFbnVtLCBkd0ZsYWdzKTsKICAgIEVudW1SZXNvdXJjZUxhbmd1YWdlc0EoIGtlcm5lbDMyX2hhbmRsZSwgKExQU1RSKVJUX1NUUklORywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUENTVFIpTE9DQUxFX0lMQU5HVUFHRSwgZW51bV9sYW5nX3Byb2NfYSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMT05HX1BUUilscGZuTG9jYWxlRW51bSk7CiAgICByZXR1cm4gVFJVRTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEVudW1TeXN0ZW1Mb2NhbGVzVyAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBFbnVtU3lzdGVtTG9jYWxlc0EuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtTG9jYWxlc1coIExPQ0FMRV9FTlVNUFJPQ1cgbHBmbkxvY2FsZUVudW0sIERXT1JEIGR3RmxhZ3MgKQp7CiAgICBUUkFDRSgiKCVwLCUwOHgpXG4iLCBscGZuTG9jYWxlRW51bSwgZHdGbGFncyk7CiAgICBFbnVtUmVzb3VyY2VMYW5ndWFnZXNXKCBrZXJuZWwzMl9oYW5kbGUsIChMUFdTVFIpUlRfU1RSSU5HLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKExQQ1dTVFIpTE9DQUxFX0lMQU5HVUFHRSwgZW51bV9sYW5nX3Byb2NfdywKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMT05HX1BUUilscGZuTG9jYWxlRW51bSk7CiAgICByZXR1cm4gVFJVRTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgVmVyTGFuZ3VhZ2VOYW1lQSAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCB0aGUgbmFtZSBvZiBhIGxhbmd1YWdlLgogKgogKiBQQVJBTVMKICogIHdMYW5nICBbSV0gTEFOR0lEIG9mIHRoZSBsYW5ndWFnZQogKiAgc3pMYW5nIFtPXSBEZXN0aW5hdGlvbiBmb3IgdGhlIGxhbmd1YWdlIG5hbWUKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVGhlIHNpemUgb2YgdGhlIGxhbmd1YWdlIG5hbWUuIElmIHN6TGFuZyBpcyBub24tTlVMTCwgaXQgaXMgZmlsbGVkCiAqICAgICAgICAgICB3aXRoIHRoZSBuYW1lLgogKiAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqCiAqLwpEV09SRCBXSU5BUEkgVmVyTGFuZ3VhZ2VOYW1lQSggRFdPUkQgd0xhbmcsIExQU1RSIHN6TGFuZywgRFdPUkQgblNpemUgKQp7CiAgICByZXR1cm4gR2V0TG9jYWxlSW5mb0EoIE1BS0VMQ0lEKHdMYW5nLCBTT1JUX0RFRkFVTFQpLCBMT0NBTEVfU0VOR0xBTkdVQUdFLCBzekxhbmcsIG5TaXplICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIFZlckxhbmd1YWdlTmFtZVcgIChLRVJORUwzMi5AKQogKgogKiBTZWUgVmVyTGFuZ3VhZ2VOYW1lQS4KICovCkRXT1JEIFdJTkFQSSBWZXJMYW5ndWFnZU5hbWVXKCBEV09SRCB3TGFuZywgTFBXU1RSIHN6TGFuZywgRFdPUkQgblNpemUgKQp7CiAgICByZXR1cm4gR2V0TG9jYWxlSW5mb1coIE1BS0VMQ0lEKHdMYW5nLCBTT1JUX0RFRkFVTFQpLCBMT0NBTEVfU0VOR0xBTkdVQUdFLCBzekxhbmcsIG5TaXplICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRTdHJpbmdUeXBlVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEdldFN0cmluZ1R5cGVBLgogKi8KQk9PTCBXSU5BUEkgR2V0U3RyaW5nVHlwZVcoIERXT1JEIHR5cGUsIExQQ1dTVFIgc3JjLCBJTlQgY291bnQsIExQV09SRCBjaGFydHlwZSApCnsKICAgIGlmIChjb3VudCA9PSAtMSkgY291bnQgPSBzdHJsZW5XKHNyYykgKyAxOwogICAgc3dpdGNoKHR5cGUpCiAgICB7CiAgICBjYXNlIENUX0NUWVBFMToKICAgICAgICB3aGlsZSAoY291bnQtLSkgKmNoYXJ0eXBlKysgPSBnZXRfY2hhcl90eXBlVyggKnNyYysrICkgJiAweGZmZjsKICAgICAgICBicmVhazsKICAgIGNhc2UgQ1RfQ1RZUEUyOgogICAgICAgIHdoaWxlIChjb3VudC0tKSAqY2hhcnR5cGUrKyA9IGdldF9jaGFyX3R5cGVXKCAqc3JjKysgKSA+PiAxMjsKICAgICAgICBicmVhazsKICAgIGNhc2UgQ1RfQ1RZUEUzOgogICAgewogICAgICAgIFdBUk4oIkNUX0NUWVBFMzogc2VtaS1zdHViLlxuIik7CiAgICAgICAgd2hpbGUgKGNvdW50LS0pCiAgICAgICAgewogICAgICAgICAgICBpbnQgYyA9ICpzcmM7CiAgICAgICAgICAgIFdPUkQgdHlwZTEsIHR5cGUzID0gMDsgLyogQzNfTk9UQVBQTElDQUJMRSAqLwoKICAgICAgICAgICAgdHlwZTEgPSBnZXRfY2hhcl90eXBlVyggKnNyYysrICkgJiAweGZmZjsKICAgICAgICAgICAgLyogdHJ5IHRvIGNvbnN0cnVjdCB0eXBlMyBmcm9tIHR5cGUxICovCiAgICAgICAgICAgIGlmKHR5cGUxICYgQzFfU1BBQ0UpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgaWYodHlwZTEgJiBDMV9BTFBIQSkgdHlwZTMgfD0gQzNfQUxQSEE7CiAgICAgICAgICAgIGlmICgoYz49MHgzMEEwKSYmKGM8PTB4MzBGRikpIHR5cGUzIHw9IEMzX0tBVEFLQU5BOwogICAgICAgICAgICBpZiAoKGM+PTB4MzA0MCkmJihjPD0weDMwOUYpKSB0eXBlMyB8PSBDM19ISVJBR0FOQTsKICAgICAgICAgICAgaWYgKChjPj0weDRFMDApJiYoYzw9MHg5RkFGKSkgdHlwZTMgfD0gQzNfSURFT0dSQVBIOwogICAgICAgICAgICBpZiAoKGM+PTB4MDYwMCkmJihjPD0weDA2RkYpKSB0eXBlMyB8PSBDM19LQVNISURBOwogICAgICAgICAgICBpZiAoKGM+PTB4MzAwMCkmJihjPD0weDMwM0YpKSB0eXBlMyB8PSBDM19TWU1CT0w7CgogICAgICAgICAgICBpZiAoKGM+PTB4RkYwMCkmJihjPD0weEZGNjApKSB0eXBlMyB8PSBDM19GVUxMV0lEVEg7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjAwKSYmKGM8PTB4RkYyMCkpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgaWYgKChjPj0weEZGM0IpJiYoYzw9MHhGRjQwKSkgdHlwZTMgfD0gQzNfU1lNQk9MOwogICAgICAgICAgICBpZiAoKGM+PTB4RkY1QikmJihjPD0weEZGNjApKSB0eXBlMyB8PSBDM19TWU1CT0w7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjIxKSYmKGM8PTB4RkYzQSkpIHR5cGUzIHw9IEMzX0FMUEhBOwogICAgICAgICAgICBpZiAoKGM+PTB4RkY0MSkmJihjPD0weEZGNUEpKSB0eXBlMyB8PSBDM19BTFBIQTsKICAgICAgICAgICAgaWYgKChjPj0weEZGRTApJiYoYzw9MHhGRkU2KSkgdHlwZTMgfD0gQzNfRlVMTFdJRFRIOwogICAgICAgICAgICBpZiAoKGM+PTB4RkZFMCkmJihjPD0weEZGRTYpKSB0eXBlMyB8PSBDM19TWU1CT0w7CgogICAgICAgICAgICBpZiAoKGM+PTB4RkY2MSkmJihjPD0weEZGREMpKSB0eXBlMyB8PSBDM19IQUxGV0lEVEg7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjYxKSYmKGM8PTB4RkY2NCkpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgaWYgKChjPj0weEZGNjUpJiYoYzw9MHhGRjlGKSkgdHlwZTMgfD0gQzNfS0FUQUtBTkE7CiAgICAgICAgICAgIGlmICgoYz49MHhGRjY1KSYmKGM8PTB4RkY5RikpIHR5cGUzIHw9IEMzX0FMUEhBOwogICAgICAgICAgICBpZiAoKGM+PTB4RkZFOCkmJihjPD0weEZGRUUpKSB0eXBlMyB8PSBDM19IQUxGV0lEVEg7CiAgICAgICAgICAgIGlmICgoYz49MHhGRkU4KSYmKGM8PTB4RkZFRSkpIHR5cGUzIHw9IEMzX1NZTUJPTDsKICAgICAgICAgICAgKmNoYXJ0eXBlKysgPSB0eXBlMzsKICAgICAgICB9CiAgICAgICAgYnJlYWs7CiAgICB9CiAgICBkZWZhdWx0OgogICAgICAgIFNldExhc3RFcnJvciggRVJST1JfSU5WQUxJRF9QQVJBTUVURVIgKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CiAgICByZXR1cm4gVFJVRTsKfQoKCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFN0cmluZ1R5cGVFeFcgICAgKEtFUk5FTDMyLkApCiAqCiAqIFNlZSBHZXRTdHJpbmdUeXBlRXhBLgogKi8KQk9PTCBXSU5BUEkgR2V0U3RyaW5nVHlwZUV4VyggTENJRCBsb2NhbGUsIERXT1JEIHR5cGUsIExQQ1dTVFIgc3JjLCBJTlQgY291bnQsIExQV09SRCBjaGFydHlwZSApCnsKICAgIC8qIGxvY2FsZSBpcyBpZ25vcmVkIGZvciBVbmljb2RlICovCiAgICByZXR1cm4gR2V0U3RyaW5nVHlwZVcoIHR5cGUsIHNyYywgY291bnQsIGNoYXJ0eXBlICk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRTdHJpbmdUeXBlQSAgICAoS0VSTkVMMzIuQCkKICoKICogR2V0IGNoYXJhY3RlcmlzdGljcyBvZiB0aGUgY2hhcmFjdGVycyBtYWtpbmcgdXAgYSBzdHJpbmcuCiAqCiAqIFBBUkFNUwogKiAgbG9jYWxlICAgW0ldIExvY2FsZSBJZCBmb3IgdGhlIHN0cmluZwogKiAgdHlwZSAgICAgW0ldIENUX0NUWVBFMSA9IGNsYXNzaWZpY2F0aW9uLCBDVF9DVFlQRTIgPSBkaXJlY3Rpb25hbGl0eSwgQ1RfQ1RZUEUzID0gdHlwb2dyYXBoaWMgaW5mbwogKiAgc3JjICAgICAgW0ldIFN0cmluZyB0byBhbmFseXNlCiAqICBjb3VudCAgICBbSV0gTGVuZ3RoIG9mIHNyYyBpbiBjaGFycywgb3IgLTEgaWYgc3JjIGlzIE5VTCB0ZXJtaW5hdGVkCiAqICBjaGFydHlwZSBbT10gRGVzdGluYXRpb24gZm9yIHRoZSBjYWxjdWxhdGVkIGNoYXJhY3RlcmlzdGljcwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLiBjaGFydHlwZSBpcyBmaWxsZWQgd2l0aCB0aGUgcmVxdWVzdGVkIGNoYXJhY3RlcmlzdGljcyBvZiBlYWNoIGNoYXIKICogICAgICAgICAgIGluIHNyYy4KICogIEZhaWx1cmU6IEZBTFNFLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCkJPT0wgV0lOQVBJIEdldFN0cmluZ1R5cGVBKCBMQ0lEIGxvY2FsZSwgRFdPUkQgdHlwZSwgTFBDU1RSIHNyYywgSU5UIGNvdW50LCBMUFdPUkQgY2hhcnR5cGUgKQp7CiAgICBVSU5UIGNwOwogICAgSU5UIGNvdW50VzsKICAgIExQV1NUUiBzcmNXOwogICAgQk9PTCByZXQgPSBGQUxTRTsKCiAgICBpZihjb3VudCA9PSAtMSkgY291bnQgPSBzdHJsZW4oc3JjKSArIDE7CgogICAgaWYgKCEoY3AgPSBnZXRfbGNpZF9jb2RlcGFnZSggbG9jYWxlICkpKQogICAgewogICAgICAgIEZJWE1FKCJGb3IgbG9jYWxlICUwNHggdXNpbmcgY3VycmVudCBBTlNJIGNvZGUgcGFnZVxuIiwgbG9jYWxlKTsKICAgICAgICBjcCA9IEdldEFDUCgpOwogICAgfQoKICAgIGNvdW50VyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIoY3AsIDAsIHNyYywgY291bnQsIE5VTEwsIDApOwogICAgaWYoKHNyY1cgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgY291bnRXICogc2l6ZW9mKFdDSEFSKSkpKQogICAgewogICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIoY3AsIDAsIHNyYywgY291bnQsIHNyY1csIGNvdW50Vyk7CiAgICAvKgogICAgICogTk9URTogdGhlIHRhcmdldCBidWZmZXIgaGFzIDEgd29yZCBmb3IgZWFjaCBDSEFSQUNURVIgaW4gdGhlIHNvdXJjZQogICAgICogc3RyaW5nLCB3aXRoIG11bHRpYnl0ZSBjaGFyYWN0ZXJzIHRoZXJlIG1heWJlIGJlIG1vcmUgYnl0ZXMgaW4gY291bnQKICAgICAqIHRoYW4gY2hhcmFjdGVyIHNwYWNlIGluIHRoZSBidWZmZXIhCiAgICAgKi8KICAgICAgICByZXQgPSBHZXRTdHJpbmdUeXBlVyh0eXBlLCBzcmNXLCBjb3VudFcsIGNoYXJ0eXBlKTsKICAgICAgICBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBzcmNXKTsKICAgIH0KICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEdldFN0cmluZ1R5cGVFeEEgICAgKEtFUk5FTDMyLkApCiAqCiAqIEdldCBjaGFyYWN0ZXJpc3RpY3Mgb2YgdGhlIGNoYXJhY3RlcnMgbWFraW5nIHVwIGEgc3RyaW5nLgogKgogKiBQQVJBTVMKICogIGxvY2FsZSAgIFtJXSBMb2NhbGUgSWQgZm9yIHRoZSBzdHJpbmcKICogIHR5cGUgICAgIFtJXSBDVF9DVFlQRTEgPSBjbGFzc2lmaWNhdGlvbiwgQ1RfQ1RZUEUyID0gZGlyZWN0aW9uYWxpdHksIENUX0NUWVBFMyA9IHR5cG9ncmFwaGljIGluZm8KICogIHNyYyAgICAgIFtJXSBTdHJpbmcgdG8gYW5hbHlzZQogKiAgY291bnQgICAgW0ldIExlbmd0aCBvZiBzcmMgaW4gY2hhcnMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgY2hhcnR5cGUgW09dIERlc3RpbmF0aW9uIGZvciB0aGUgY2FsY3VsYXRlZCBjaGFyYWN0ZXJpc3RpY3MKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4gY2hhcnR5cGUgaXMgZmlsbGVkIHdpdGggdGhlIHJlcXVlc3RlZCBjaGFyYWN0ZXJpc3RpY3Mgb2YgZWFjaCBjaGFyCiAqICAgICAgICAgICBpbiBzcmMuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBHZXRTdHJpbmdUeXBlRXhBKCBMQ0lEIGxvY2FsZSwgRFdPUkQgdHlwZSwgTFBDU1RSIHNyYywgSU5UIGNvdW50LCBMUFdPUkQgY2hhcnR5cGUgKQp7CiAgICByZXR1cm4gR2V0U3RyaW5nVHlwZUEobG9jYWxlLCB0eXBlLCBzcmMsIGNvdW50LCBjaGFydHlwZSk7Cn0KCgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgTENNYXBTdHJpbmdXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgTENNYXBTdHJpbmdBLgogKi8KSU5UIFdJTkFQSSBMQ01hcFN0cmluZ1coTENJRCBsY2lkLCBEV09SRCBmbGFncywgTFBDV1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgIExQV1NUUiBkc3QsIElOVCBkc3RsZW4pCnsKICAgIExQV1NUUiBkc3RfcHRyOwoKICAgIGlmICghc3JjIHx8ICFzcmNsZW4gfHwgZHN0bGVuIDwgMCkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIC8qIG11dHVhbGx5IGV4Y2x1c2l2ZSBmbGFncyAqLwogICAgaWYgKChmbGFncyAmIChMQ01BUF9MT1dFUkNBU0UgfCBMQ01BUF9VUFBFUkNBU0UpKSA9PSAoTENNQVBfTE9XRVJDQVNFIHwgTENNQVBfVVBQRVJDQVNFKSB8fAogICAgICAgIChmbGFncyAmIChMQ01BUF9ISVJBR0FOQSB8IExDTUFQX0tBVEFLQU5BKSkgPT0gKExDTUFQX0hJUkFHQU5BIHwgTENNQVBfS0FUQUtBTkEpIHx8CiAgICAgICAgKGZsYWdzICYgKExDTUFQX0hBTEZXSURUSCB8IExDTUFQX0ZVTExXSURUSCkpID09IChMQ01BUF9IQUxGV0lEVEggfCBMQ01BUF9GVUxMV0lEVEgpIHx8CiAgICAgICAgKGZsYWdzICYgKExDTUFQX1RSQURJVElPTkFMX0NISU5FU0UgfCBMQ01BUF9TSU1QTElGSUVEX0NISU5FU0UpKSA9PSAoTENNQVBfVFJBRElUSU9OQUxfQ0hJTkVTRSB8IExDTUFQX1NJTVBMSUZJRURfQ0hJTkVTRSkpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmICghZHN0bGVuKSBkc3QgPSBOVUxMOwoKICAgIGxjaWQgPSBDb252ZXJ0RGVmYXVsdExvY2FsZShsY2lkKTsKCiAgICBpZiAoZmxhZ3MgJiBMQ01BUF9TT1JUS0VZKQogICAgewogICAgICAgIElOVCByZXQ7CiAgICAgICAgaWYgKHNyYyA9PSBkc3QpCiAgICAgICAgewogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgICAgIHJldHVybiAwOwogICAgICAgIH0KCiAgICAgICAgaWYgKHNyY2xlbiA8IDApIHNyY2xlbiA9IHN0cmxlblcoc3JjKTsKCiAgICAgICAgVFJBQ0UoIigweCUwNHgsMHglMDh4LCVzLCVkLCVwLCVkKVxuIiwKICAgICAgICAgICAgICBsY2lkLCBmbGFncywgZGVidWdzdHJfd24oc3JjLCBzcmNsZW4pLCBzcmNsZW4sIGRzdCwgZHN0bGVuKTsKCiAgICAgICAgcmV0ID0gd2luZV9nZXRfc29ydGtleShmbGFncywgc3JjLCBzcmNsZW4sIChjaGFyICopZHN0LCBkc3RsZW4pOwogICAgICAgIGlmIChyZXQgPT0gMCkKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOU1VGRklDSUVOVF9CVUZGRVIpOwogICAgICAgIHJldHVybiByZXQ7CiAgICB9CgogICAgLyogU09SVF9TVFJJTkdTT1JUIG11c3QgYmUgdXNlZCBleGNsdXNpdmVseSB3aXRoIExDTUFQX1NPUlRLRVkgKi8KICAgIGlmIChmbGFncyAmIFNPUlRfU1RSSU5HU09SVCkKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYgKHNyY2xlbiA8IDApIHNyY2xlbiA9IHN0cmxlblcoc3JjKSArIDE7CgogICAgVFJBQ0UoIigweCUwNHgsMHglMDh4LCVzLCVkLCVwLCVkKVxuIiwKICAgICAgICAgIGxjaWQsIGZsYWdzLCBkZWJ1Z3N0cl93bihzcmMsIHNyY2xlbiksIHNyY2xlbiwgZHN0LCBkc3RsZW4pOwoKICAgIGlmICghZHN0KSAvKiByZXR1cm4gcmVxdWlyZWQgc3RyaW5nIGxlbmd0aCAqLwogICAgewogICAgICAgIElOVCBsZW47CgogICAgICAgIGZvciAobGVuID0gMDsgc3JjbGVuOyBzcmMrKywgc3JjbGVuLS0pCiAgICAgICAgewogICAgICAgICAgICBXQ0hBUiB3Y2ggPSAqc3JjOwogICAgICAgICAgICAvKiB0ZXN0cyBzaG93IHRoYXQgd2luMmsganVzdCBpZ25vcmVzIE5PUk1fSUdOT1JFTk9OU1BBQ0UsCiAgICAgICAgICAgICAqIGFuZCBza2lwcyB3aGl0ZSBzcGFjZSBhbmQgcHVuY3R1YXRpb24gY2hhcmFjdGVycyBmb3IKICAgICAgICAgICAgICogTk9STV9JR05PUkVTWU1CT0xTLgogICAgICAgICAgICAgKi8KICAgICAgICAgICAgaWYgKChmbGFncyAmIE5PUk1fSUdOT1JFU1lNQk9MUykgJiYgKGdldF9jaGFyX3R5cGVXKHdjaCkgJiAoQzFfUFVOQ1QgfCBDMV9TUEFDRSkpKQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgIGxlbisrOwogICAgICAgIH0KICAgICAgICByZXR1cm4gbGVuOwogICAgfQoKICAgIGlmIChmbGFncyAmIExDTUFQX1VQUEVSQ0FTRSkKICAgIHsKICAgICAgICBmb3IgKGRzdF9wdHIgPSBkc3Q7IHNyY2xlbiAmJiBkc3RsZW47IHNyYysrLCBzcmNsZW4tLSkKICAgICAgICB7CiAgICAgICAgICAgIFdDSEFSIHdjaCA9ICpzcmM7CiAgICAgICAgICAgIGlmICgoZmxhZ3MgJiBOT1JNX0lHTk9SRVNZTUJPTFMpICYmIChnZXRfY2hhcl90eXBlVyh3Y2gpICYgKEMxX1BVTkNUIHwgQzFfU1BBQ0UpKSkKICAgICAgICAgICAgICAgIGNvbnRpbnVlOwogICAgICAgICAgICAqZHN0X3B0cisrID0gdG91cHBlclcod2NoKTsKICAgICAgICAgICAgZHN0bGVuLS07CiAgICAgICAgfQogICAgfQogICAgZWxzZSBpZiAoZmxhZ3MgJiBMQ01BUF9MT1dFUkNBU0UpCiAgICB7CiAgICAgICAgZm9yIChkc3RfcHRyID0gZHN0OyBzcmNsZW4gJiYgZHN0bGVuOyBzcmMrKywgc3JjbGVuLS0pCiAgICAgICAgewogICAgICAgICAgICBXQ0hBUiB3Y2ggPSAqc3JjOwogICAgICAgICAgICBpZiAoKGZsYWdzICYgTk9STV9JR05PUkVTWU1CT0xTKSAmJiAoZ2V0X2NoYXJfdHlwZVcod2NoKSAmIChDMV9QVU5DVCB8IEMxX1NQQUNFKSkpCiAgICAgICAgICAgICAgICBjb250aW51ZTsKICAgICAgICAgICAgKmRzdF9wdHIrKyA9IHRvbG93ZXJXKHdjaCk7CiAgICAgICAgICAgIGRzdGxlbi0tOwogICAgICAgIH0KICAgIH0KICAgIGVsc2UKICAgIHsKICAgICAgICBpZiAoc3JjID09IGRzdCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIGZvciAoZHN0X3B0ciA9IGRzdDsgc3JjbGVuICYmIGRzdGxlbjsgc3JjKyssIHNyY2xlbi0tKQogICAgICAgIHsKICAgICAgICAgICAgV0NIQVIgd2NoID0gKnNyYzsKICAgICAgICAgICAgaWYgKChmbGFncyAmIE5PUk1fSUdOT1JFU1lNQk9MUykgJiYgKGdldF9jaGFyX3R5cGVXKHdjaCkgJiAoQzFfUFVOQ1QgfCBDMV9TUEFDRSkpKQogICAgICAgICAgICAgICAgY29udGludWU7CiAgICAgICAgICAgICpkc3RfcHRyKysgPSB3Y2g7CiAgICAgICAgICAgIGRzdGxlbi0tOwogICAgICAgIH0KICAgIH0KCiAgICBpZiAoc3JjbGVuKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICByZXR1cm4gZHN0X3B0ciAtIGRzdDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIExDTWFwU3RyaW5nQSAgICAoS0VSTkVMMzIuQCkKICoKICogTWFwIGNoYXJhY3RlcnMgaW4gYSBsb2NhbGUgc2Vuc2l0aXZlIHN0cmluZy4KICoKICogUEFSQU1TCiAqICBsY2lkICAgW0ldIExDSUQgZm9yIHRoZSBjb252ZXJzaW9uLgogKiAgZmxhZ3MgIFtJXSBGbGFncyBjb250cm9sbGluZyB0aGUgbWFwcGluZyAoTENNQVBfIGNvbnN0YW50cyBmcm9tICJ3aW5ubHMuaCIpLgogKiAgc3JjICAgIFtJXSBTdHJpbmcgdG8gbWFwCiAqICBzcmNsZW4gW0ldIExlbmd0aCBvZiBzcmMgaW4gY2hhcnMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgZHN0ICAgIFtPXSBEZXN0aW5hdGlvbiBmb3IgbWFwcGVkIHN0cmluZwogKiAgZHN0bGVuIFtJXSBMZW5ndGggb2YgZHN0IGluIGNoYXJhY3RlcnMKICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVGhlIGxlbmd0aCBvZiB0aGUgbWFwcGVkIHN0cmluZyBpbiBkc3QsIGluY2x1ZGluZyB0aGUgTlVMIHRlcm1pbmF0b3IuCiAqICBGYWlsdXJlOiAwLiBVc2UgR2V0TGFzdEVycm9yKCkgdG8gZGV0ZXJtaW5lIHRoZSBjYXVzZS4KICovCklOVCBXSU5BUEkgTENNYXBTdHJpbmdBKExDSUQgbGNpZCwgRFdPUkQgZmxhZ3MsIExQQ1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgIExQU1RSIGRzdCwgSU5UIGRzdGxlbikKewogICAgV0NIQVIgKmJ1ZlcgPSBOdEN1cnJlbnRUZWIoKS0+U3RhdGljVW5pY29kZUJ1ZmZlcjsKICAgIExQV1NUUiBzcmNXLCBkc3RXOwogICAgSU5UIHJldCA9IDAsIHNyY2xlblcsIGRzdGxlblc7CiAgICBVSU5UIGxvY2FsZV9jcCA9IENQX0FDUDsKCiAgICBpZiAoIXNyYyB8fCAhc3JjbGVuIHx8IGRzdGxlbiA8IDApCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICBpZiAoIShmbGFncyAmIExPQ0FMRV9VU0VfQ1BfQUNQKSkgbG9jYWxlX2NwID0gZ2V0X2xjaWRfY29kZXBhZ2UoIGxjaWQgKTsKCiAgICBzcmNsZW5XID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHNyYywgc3JjbGVuLCBidWZXLCAyNjApOwogICAgaWYgKHNyY2xlblcpCiAgICAgICAgc3JjVyA9IGJ1Zlc7CiAgICBlbHNlCiAgICB7CiAgICAgICAgc3JjbGVuVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzcmMsIHNyY2xlbiwgTlVMTCwgMCk7CiAgICAgICAgc3JjVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBzcmNsZW5XICogc2l6ZW9mKFdDSEFSKSk7CiAgICAgICAgaWYgKCFzcmNXKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzcmMsIHNyY2xlbiwgc3JjVywgc3JjbGVuVyk7CiAgICB9CgogICAgaWYgKGZsYWdzICYgTENNQVBfU09SVEtFWSkKICAgIHsKICAgICAgICBpZiAoc3JjID09IGRzdCkKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKICAgICAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CiAgICAgICAgfQogICAgICAgIHJldCA9IHdpbmVfZ2V0X3NvcnRrZXkoZmxhZ3MsIHNyY1csIHNyY2xlblcsIGRzdCwgZHN0bGVuKTsKICAgICAgICBpZiAocmV0ID09IDApCiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlNVRkZJQ0lFTlRfQlVGRkVSKTsKICAgICAgICBnb3RvIG1hcF9zdHJpbmdfZXhpdDsKICAgIH0KCiAgICBpZiAoZmxhZ3MgJiBTT1JUX1NUUklOR1NPUlQpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIGdvdG8gbWFwX3N0cmluZ19leGl0OwogICAgfQoKICAgIGRzdGxlblcgPSBMQ01hcFN0cmluZ1cobGNpZCwgZmxhZ3MsIHNyY1csIHNyY2xlblcsIE5VTEwsIDApOwogICAgaWYgKCFkc3RsZW5XKQogICAgICAgIGdvdG8gbWFwX3N0cmluZ19leGl0OwoKICAgIGRzdFcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgZHN0bGVuVyAqIHNpemVvZihXQ0hBUikpOwogICAgaWYgKCFkc3RXKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgZ290byBtYXBfc3RyaW5nX2V4aXQ7CiAgICB9CgogICAgTENNYXBTdHJpbmdXKGxjaWQsIGZsYWdzLCBzcmNXLCBzcmNsZW5XLCBkc3RXLCBkc3RsZW5XKTsKICAgIHJldCA9IFdpZGVDaGFyVG9NdWx0aUJ5dGUobG9jYWxlX2NwLCAwLCBkc3RXLCBkc3RsZW5XLCBkc3QsIGRzdGxlbiwgTlVMTCwgTlVMTCk7CiAgICBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBkc3RXKTsKCm1hcF9zdHJpbmdfZXhpdDoKICAgIGlmIChzcmNXICE9IGJ1ZlcpIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEZvbGRTdHJpbmdBICAgIChLRVJORUwzMi5AKQogKgogKiBNYXAgY2hhcmFjdGVycyBpbiBhIHN0cmluZy4KICoKICogUEFSQU1TCiAqICBkd0ZsYWdzIFtJXSBGbGFncyBjb250cm9sbGluZyBjaGFycyB0byBtYXAgKE1BUF8gY29uc3RhbnRzIGZyb20gIndpbm5scy5oIikKICogIHNyYyAgICAgW0ldIFN0cmluZyB0byBtYXAKICogIHNyY2xlbiAgW0ldIExlbmd0aCBvZiBzcmMsIG9yIC0xIGlmIHNyYyBpcyBOVUwgdGVybWluYXRlZAogKiAgZHN0ICAgICBbT10gRGVzdGluYXRpb24gZm9yIG1hcHBlZCBzdHJpbmcKICogIGRzdGxlbiAgW0ldIExlbmd0aCBvZiBkc3QsIG9yIDAgdG8gZmluZCB0aGUgcmVxdWlyZWQgbGVuZ3RoIGZvciB0aGUgbWFwcGVkIHN0cmluZwogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUaGUgbGVuZ3RoIG9mIHRoZSBzdHJpbmcgd3JpdHRlbiB0byBkc3QsIGluY2x1ZGluZyB0aGUgdGVybWluYXRpbmcgTlVMLiBJZgogKiAgICAgICAgICAgZHN0bGVuIGlzIDAsIHRoZSB2YWx1ZSByZXR1cm5lZCBpcyB0aGUgc2FtZSwgYnV0IG5vdGhpbmcgaXMgd3JpdHRlbiB0byBkc3QsCiAqICAgICAgICAgICBhbmQgZHN0IG1heSBiZSBOVUxMLgogKiAgRmFpbHVyZTogMC4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpJTlQgV0lOQVBJIEZvbGRTdHJpbmdBKERXT1JEIGR3RmxhZ3MsIExQQ1NUUiBzcmMsIElOVCBzcmNsZW4sCiAgICAgICAgICAgICAgICAgICAgICAgTFBTVFIgZHN0LCBJTlQgZHN0bGVuKQp7CiAgICBJTlQgcmV0ID0gMCwgc3JjbGVuVyA9IDA7CiAgICBXQ0hBUiAqc3JjVyA9IE5VTEwsICpkc3RXID0gTlVMTDsKCiAgICBpZiAoIXNyYyB8fCAhc3JjbGVuIHx8IGRzdGxlbiA8IDAgfHwgKGRzdGxlbiAmJiAhZHN0KSB8fCBzcmMgPT0gZHN0KQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgc3JjbGVuVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIoQ1BfQUNQLCBkd0ZsYWdzICYgTUFQX0NPTVBPU0lURSA/IE1CX0NPTVBPU0lURSA6IDAsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzcmMsIHNyY2xlbiwgTlVMTCwgMCk7CiAgICBzcmNXID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY2xlblcgKiBzaXplb2YoV0NIQVIpKTsKCiAgICBpZiAoIXNyY1cpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICBnb3RvIEZvbGRTdHJpbmdBX2V4aXQ7CiAgICB9CgogICAgTXVsdGlCeXRlVG9XaWRlQ2hhcihDUF9BQ1AsIGR3RmxhZ3MgJiBNQVBfQ09NUE9TSVRFID8gTUJfQ09NUE9TSVRFIDogMCwKICAgICAgICAgICAgICAgICAgICAgICAgc3JjLCBzcmNsZW4sIHNyY1csIHNyY2xlblcpOwoKICAgIGR3RmxhZ3MgPSAoZHdGbGFncyAmIH5NQVBfUFJFQ09NUE9TRUQpIHwgTUFQX0ZPTERDWk9ORTsKCiAgICByZXQgPSBGb2xkU3RyaW5nVyhkd0ZsYWdzLCBzcmNXLCBzcmNsZW5XLCBOVUxMLCAwKTsKICAgIGlmIChyZXQgJiYgZHN0bGVuKQogICAgewogICAgICAgIGRzdFcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgcmV0ICogc2l6ZW9mKFdDSEFSKSk7CgogICAgICAgIGlmICghZHN0VykKICAgICAgICB7CiAgICAgICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9OT1RfRU5PVUdIX01FTU9SWSk7CiAgICAgICAgICAgIGdvdG8gRm9sZFN0cmluZ0FfZXhpdDsKICAgICAgICB9CgogICAgICAgIHJldCA9IEZvbGRTdHJpbmdXKGR3RmxhZ3MsIHNyY1csIHNyY2xlblcsIGRzdFcsIHJldCk7CiAgICAgICAgaWYgKCFXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgZHN0VywgcmV0LCBkc3QsIGRzdGxlbiwgTlVMTCwgTlVMTCkpCiAgICAgICAgewogICAgICAgICAgICByZXQgPSAwOwogICAgICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUik7CiAgICAgICAgfQogICAgfQoKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGRzdFcpOwoKRm9sZFN0cmluZ0FfZXhpdDoKICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHNyY1cpOwogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEZvbGRTdHJpbmdXICAgIChLRVJORUwzMi5AKQogKgogKiBTZWUgRm9sZFN0cmluZ0EuCiAqLwpJTlQgV0lOQVBJIEZvbGRTdHJpbmdXKERXT1JEIGR3RmxhZ3MsIExQQ1dTVFIgc3JjLCBJTlQgc3JjbGVuLAogICAgICAgICAgICAgICAgICAgICAgIExQV1NUUiBkc3QsIElOVCBkc3RsZW4pCnsKICAgIGludCByZXQ7CgogICAgc3dpdGNoIChkd0ZsYWdzICYgKE1BUF9DT01QT1NJVEV8TUFQX1BSRUNPTVBPU0VEfE1BUF9FWFBBTkRfTElHQVRVUkVTKSkKICAgIHsKICAgIGNhc2UgMDoKICAgICAgICBpZiAoZHdGbGFncykKICAgICAgICAgIGJyZWFrOwogICAgICAgIC8qIEZhbGwgdGhyb3VnaCBmb3IgZHdGbGFncyA9PSAwICovCiAgICBjYXNlIE1BUF9QUkVDT01QT1NFRHxNQVBfQ09NUE9TSVRFOgogICAgY2FzZSBNQVBfUFJFQ09NUE9TRUR8TUFQX0VYUEFORF9MSUdBVFVSRVM6CiAgICBjYXNlIE1BUF9DT01QT1NJVEV8TUFQX0VYUEFORF9MSUdBVFVSRVM6CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIGlmICghc3JjIHx8ICFzcmNsZW4gfHwgZHN0bGVuIDwgMCB8fCAoZHN0bGVuICYmICFkc3QpIHx8IHNyYyA9PSBkc3QpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gMDsKICAgIH0KCiAgICByZXQgPSB3aW5lX2ZvbGRfc3RyaW5nKGR3RmxhZ3MsIHNyYywgc3JjbGVuLCBkc3QsIGRzdGxlbik7CiAgICBpZiAoIXJldCkKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5TVUZGSUNJRU5UX0JVRkZFUik7CiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBDb21wYXJlU3RyaW5nVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIENvbXBhcmVTdHJpbmdBLgogKi8KSU5UIFdJTkFQSSBDb21wYXJlU3RyaW5nVyhMQ0lEIGxjaWQsIERXT1JEIHN0eWxlLAogICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1dTVFIgc3RyMSwgSU5UIGxlbjEsIExQQ1dTVFIgc3RyMiwgSU5UIGxlbjIpCnsKICAgIElOVCByZXQ7CgogICAgaWYgKCFzdHIxIHx8ICFzdHIyKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIDA7CiAgICB9CgogICAgaWYoIHN0eWxlICYgfihOT1JNX0lHTk9SRUNBU0V8Tk9STV9JR05PUkVOT05TUEFDRXxOT1JNX0lHTk9SRVNZTUJPTFN8CiAgICAgICAgU09SVF9TVFJJTkdTT1JUfE5PUk1fSUdOT1JFS0FOQVRZUEV8Tk9STV9JR05PUkVXSURUSHxMT0NBTEVfVVNFX0NQX0FDUHwweDEwMDAwMDAwKSApCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiAwOwogICAgfQoKICAgIC8qIHRoaXMgc3R5bGUgaXMgcmVsYXRlZCB0byBkaWFjcml0aWNzIGluIEFyYWJpYywgSmFwYW5lc2UsIGFuZCBIZWJyZXcgKi8KICAgIGlmIChzdHlsZSAmIDB4MTAwMDAwMDApCiAgICAgICAgV0FSTigiSWdub3JpbmcgdW5rbm93biBzdHlsZSAweDEwMDAwMDAwXG4iKTsKCiAgICBpZiAobGVuMSA8IDApIGxlbjEgPSBzdHJsZW5XKHN0cjEpOwogICAgaWYgKGxlbjIgPCAwKSBsZW4yID0gc3RybGVuVyhzdHIyKTsKCiAgICByZXQgPSB3aW5lX2NvbXBhcmVfc3RyaW5nKHN0eWxlLCBzdHIxLCBsZW4xLCBzdHIyLCBsZW4yKTsKCiAgICBpZiAocmV0KSAvKiBuZWVkIHRvIHRyYW5zbGF0ZSByZXN1bHQgKi8KICAgICAgICByZXR1cm4gKHJldCA8IDApID8gQ1NUUl9MRVNTX1RIQU4gOiBDU1RSX0dSRUFURVJfVEhBTjsKICAgIHJldHVybiBDU1RSX0VRVUFMOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBDb21wYXJlU3RyaW5nQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ29tcGFyZSB0d28gbG9jYWxlIHNlbnNpdGl2ZSBzdHJpbmdzLgogKgogKiBQQVJBTVMKICogIGxjaWQgIFtJXSBMQ0lEIGZvciB0aGUgY29tcGFyaXNvbgogKiAgc3R5bGUgW0ldIEZsYWdzIGZvciB0aGUgY29tcGFyaXNvbiAoTk9STV8gY29uc3RhbnRzIGZyb20gIndpbm5scy5oIikuCiAqICBzdHIxICBbSV0gRmlyc3Qgc3RyaW5nIHRvIGNvbXBhcmUKICogIGxlbjEgIFtJXSBMZW5ndGggb2Ygc3RyMSwgb3IgLTEgaWYgc3RyMSBpcyBOVUwgdGVybWluYXRlZAogKiAgc3RyMiAgW0ldIFNlY29uZCBzdHJpbmcgdG8gY29tcGFyZQogKiAgbGVuMiAgW0ldIExlbmd0aCBvZiBzdHIyLCBvciAtMSBpZiBzdHIyIGlzIE5VTCB0ZXJtaW5hdGVkCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IENTVFJfTEVTU19USEFOLCBDU1RSX0VRVUFMIG9yIENTVFJfR1JFQVRFUl9USEFOIGRlcGVuZGluZyBvbiB3aGV0aGVyCiAqICAgICAgICAgICBzdHIyIGlzIGxlc3MgdGhhbiwgZXF1YWwgdG8gb3IgZ3JlYXRlciB0aGFuIHN0cjEgcmVzcGVjdGl2ZWx5LgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KSU5UIFdJTkFQSSBDb21wYXJlU3RyaW5nQShMQ0lEIGxjaWQsIERXT1JEIHN0eWxlLAogICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1NUUiBzdHIxLCBJTlQgbGVuMSwgTFBDU1RSIHN0cjIsIElOVCBsZW4yKQp7CiAgICBXQ0hBUiAqYnVmMVcgPSBOdEN1cnJlbnRUZWIoKS0+U3RhdGljVW5pY29kZUJ1ZmZlcjsKICAgIFdDSEFSICpidWYyVyA9IGJ1ZjFXICsgMTMwOwogICAgTFBXU1RSIHN0cjFXLCBzdHIyVzsKICAgIElOVCBsZW4xVywgbGVuMlcsIHJldDsKICAgIFVJTlQgbG9jYWxlX2NwID0gQ1BfQUNQOwoKICAgIGlmICghc3RyMSB8fCAhc3RyMikKICAgIHsKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9QQVJBTUVURVIpOwogICAgICAgIHJldHVybiAwOwogICAgfQogICAgaWYgKGxlbjEgPCAwKSBsZW4xID0gc3RybGVuKHN0cjEpOwogICAgaWYgKGxlbjIgPCAwKSBsZW4yID0gc3RybGVuKHN0cjIpOwoKICAgIGlmICghKHN0eWxlICYgTE9DQUxFX1VTRV9DUF9BQ1ApKSBsb2NhbGVfY3AgPSBnZXRfbGNpZF9jb2RlcGFnZSggbGNpZCApOwoKICAgIGxlbjFXID0gTXVsdGlCeXRlVG9XaWRlQ2hhcihsb2NhbGVfY3AsIDAsIHN0cjEsIGxlbjEsIGJ1ZjFXLCAxMzApOwogICAgaWYgKGxlbjFXKQogICAgICAgIHN0cjFXID0gYnVmMVc7CiAgICBlbHNlCiAgICB7CiAgICAgICAgbGVuMVcgPSBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMSwgbGVuMSwgTlVMTCwgMCk7CiAgICAgICAgc3RyMVcgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgbGVuMVcgKiBzaXplb2YoV0NIQVIpKTsKICAgICAgICBpZiAoIXN0cjFXKQogICAgICAgIHsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIxLCBsZW4xLCBzdHIxVywgbGVuMVcpOwogICAgfQogICAgbGVuMlcgPSBNdWx0aUJ5dGVUb1dpZGVDaGFyKGxvY2FsZV9jcCwgMCwgc3RyMiwgbGVuMiwgYnVmMlcsIDEzMCk7CiAgICBpZiAobGVuMlcpCiAgICAgICAgc3RyMlcgPSBidWYyVzsKICAgIGVsc2UKICAgIHsKICAgICAgICBsZW4yVyA9IE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIyLCBsZW4yLCBOVUxMLCAwKTsKICAgICAgICBzdHIyVyA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBsZW4yVyAqIHNpemVvZihXQ0hBUikpOwogICAgICAgIGlmICghc3RyMlcpCiAgICAgICAgewogICAgICAgICAgICBpZiAoc3RyMVcgIT0gYnVmMVcpIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHN0cjFXKTsKICAgICAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX05PVF9FTk9VR0hfTUVNT1JZKTsKICAgICAgICAgICAgcmV0dXJuIDA7CiAgICAgICAgfQogICAgICAgIE11bHRpQnl0ZVRvV2lkZUNoYXIobG9jYWxlX2NwLCAwLCBzdHIyLCBsZW4yLCBzdHIyVywgbGVuMlcpOwogICAgfQoKICAgIHJldCA9IENvbXBhcmVTdHJpbmdXKGxjaWQsIHN0eWxlLCBzdHIxVywgbGVuMVcsIHN0cjJXLCBsZW4yVyk7CgogICAgaWYgKHN0cjFXICE9IGJ1ZjFXKSBIZWFwRnJlZShHZXRQcm9jZXNzSGVhcCgpLCAwLCBzdHIxVyk7CiAgICBpZiAoc3RyMlcgIT0gYnVmMlcpIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIHN0cjJXKTsKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBsc3RyY21wICAgICAoS0VSTkVMMzIuQCkKICogICAgICAgICAgIGxzdHJjbXBBICAgIChLRVJORUwzMi5AKQogKgogKiBDb21wYXJlIHR3byBzdHJpbmdzIHVzaW5nIHRoZSBjdXJyZW50IHRocmVhZCBsb2NhbGUuCiAqCiAqIFBBUkFNUwogKiAgc3RyMSAgW0ldIEZpcnN0IHN0cmluZyB0byBjb21wYXJlCiAqICBzdHIyICBbSV0gU2Vjb25kIHN0cmluZyB0byBjb21wYXJlCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IEEgbnVtYmVyIGxlc3MgdGhhbiwgZXF1YWwgdG8gb3IgZ3JlYXRlciB0aGFuIDAgZGVwZW5kaW5nIG9uIHdoZXRoZXIKICogICAgICAgICAgIHN0cjIgaXMgbGVzcyB0aGFuLCBlcXVhbCB0byBvciBncmVhdGVyIHRoYW4gc3RyMSByZXNwZWN0aXZlbHkuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwppbnQgV0lOQVBJIGxzdHJjbXBBKExQQ1NUUiBzdHIxLCBMUENTVFIgc3RyMikKewogICAgaW50IHJldDsKICAgIAogICAgaWYgKChzdHIxID09IE5VTEwpICYmIChzdHIyID09IE5VTEwpKSByZXR1cm4gMDsKICAgIGlmIChzdHIxID09IE5VTEwpIHJldHVybiAtMTsKICAgIGlmIChzdHIyID09IE5VTEwpIHJldHVybiAxOwoKICAgIHJldCA9IENvbXBhcmVTdHJpbmdBKEdldFRocmVhZExvY2FsZSgpLCBMT0NBTEVfVVNFX0NQX0FDUCwgc3RyMSwgLTEsIHN0cjIsIC0xKTsKICAgIGlmIChyZXQpIHJldCAtPSAyOwogICAgCiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgbHN0cmNtcGkgICAgIChLRVJORUwzMi5AKQogKiAgICAgICAgICAgbHN0cmNtcGlBICAgIChLRVJORUwzMi5AKQogKgogKiBDb21wYXJlIHR3byBzdHJpbmdzIHVzaW5nIHRoZSBjdXJyZW50IHRocmVhZCBsb2NhbGUsIGlnbm9yaW5nIGNhc2UuCiAqCiAqIFBBUkFNUwogKiAgc3RyMSAgW0ldIEZpcnN0IHN0cmluZyB0byBjb21wYXJlCiAqICBzdHIyICBbSV0gU2Vjb25kIHN0cmluZyB0byBjb21wYXJlCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IEEgbnVtYmVyIGxlc3MgdGhhbiwgZXF1YWwgdG8gb3IgZ3JlYXRlciB0aGFuIDAgZGVwZW5kaW5nIG9uIHdoZXRoZXIKICogICAgICAgICAgIHN0cjIgaXMgbGVzcyB0aGFuLCBlcXVhbCB0byBvciBncmVhdGVyIHRoYW4gc3RyMSByZXNwZWN0aXZlbHkuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwppbnQgV0lOQVBJIGxzdHJjbXBpQShMUENTVFIgc3RyMSwgTFBDU1RSIHN0cjIpCnsKICAgIGludCByZXQ7CiAgICAKICAgIGlmICgoc3RyMSA9PSBOVUxMKSAmJiAoc3RyMiA9PSBOVUxMKSkgcmV0dXJuIDA7CiAgICBpZiAoc3RyMSA9PSBOVUxMKSByZXR1cm4gLTE7CiAgICBpZiAoc3RyMiA9PSBOVUxMKSByZXR1cm4gMTsKCiAgICByZXQgPSBDb21wYXJlU3RyaW5nQShHZXRUaHJlYWRMb2NhbGUoKSwgTk9STV9JR05PUkVDQVNFfExPQ0FMRV9VU0VfQ1BfQUNQLCBzdHIxLCAtMSwgc3RyMiwgLTEpOwogICAgaWYgKHJldCkgcmV0IC09IDI7CiAgICAKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBsc3RyY21wVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIGxzdHJjbXBBLgogKi8KaW50IFdJTkFQSSBsc3RyY21wVyhMUENXU1RSIHN0cjEsIExQQ1dTVFIgc3RyMikKewogICAgaW50IHJldDsKCiAgICBpZiAoKHN0cjEgPT0gTlVMTCkgJiYgKHN0cjIgPT0gTlVMTCkpIHJldHVybiAwOwogICAgaWYgKHN0cjEgPT0gTlVMTCkgcmV0dXJuIC0xOwogICAgaWYgKHN0cjIgPT0gTlVMTCkgcmV0dXJuIDE7CgogICAgcmV0ID0gQ29tcGFyZVN0cmluZ1coR2V0VGhyZWFkTG9jYWxlKCksIDAsIHN0cjEsIC0xLCBzdHIyLCAtMSk7CiAgICBpZiAocmV0KSByZXQgLT0gMjsKICAgIAogICAgcmV0dXJuIHJldDsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIGxzdHJjbXBpVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIGxzdHJjbXBpQS4KICovCmludCBXSU5BUEkgbHN0cmNtcGlXKExQQ1dTVFIgc3RyMSwgTFBDV1NUUiBzdHIyKQp7CiAgICBpbnQgcmV0OwogICAgCiAgICBpZiAoKHN0cjEgPT0gTlVMTCkgJiYgKHN0cjIgPT0gTlVMTCkpIHJldHVybiAwOwogICAgaWYgKHN0cjEgPT0gTlVMTCkgcmV0dXJuIC0xOwogICAgaWYgKHN0cjIgPT0gTlVMTCkgcmV0dXJuIDE7CgogICAgcmV0ID0gQ29tcGFyZVN0cmluZ1coR2V0VGhyZWFkTG9jYWxlKCksIE5PUk1fSUdOT1JFQ0FTRSwgc3RyMSwgLTEsIHN0cjIsIC0xKTsKICAgIGlmIChyZXQpIHJldCAtPSAyOwogICAgCiAgICByZXR1cm4gcmV0Owp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqCQlMT0NBTEVfSW5pdAogKi8Kdm9pZCBMT0NBTEVfSW5pdCh2b2lkKQp7CiAgICBleHRlcm4gdm9pZCBfX3dpbmVfaW5pdF9jb2RlcGFnZXMoIGNvbnN0IHVuaW9uIGNwdGFibGUgKmFuc2lfY3AsIGNvbnN0IHVuaW9uIGNwdGFibGUgKm9lbV9jcCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdW5pb24gY3B0YWJsZSAqdW5peF9jcCApOwoKICAgIFVJTlQgYW5zaV9jcCA9IDEyNTIsIG9lbV9jcCA9IDQzNywgbWFjX2NwID0gMTAwMDAsIHVuaXhfY3A7CgojaWZkZWYgX19BUFBMRV9fCiAgICAvKiBNYWNPUyBkb2Vzbid0IHNldCB0aGUgbG9jYWxlIGVudmlyb25tZW50IHZhcmlhYmxlcyBzbyB3ZSBoYXZlIHRvIGRvIGl0IG91cnNlbHZlcyAqLwogICAgQ0ZBcnJheVJlZiBwcmVmZXJyZWRfbG9jYWxlcywgYWxsX2xvY2FsZXM7CiAgICBDRlN0cmluZ1JlZiB1c2VyX2xhbmd1YWdlX3N0cmluZ19yZWYgPSBOVUxMOwogICAgY2hhciB1c2VyX2xvY2FsZVs1MF07CgogICAgQ0ZMb2NhbGVSZWYgdXNlcl9sb2NhbGVfcmVmID0gQ0ZMb2NhbGVDb3B5Q3VycmVudCgpOwogICAgQ0ZTdHJpbmdSZWYgdXNlcl9sb2NhbGVfc3RyaW5nX3JlZiA9IENGTG9jYWxlR2V0SWRlbnRpZmllciggdXNlcl9sb2NhbGVfcmVmICk7CgogICAgQ0ZTdHJpbmdHZXRDU3RyaW5nKCB1c2VyX2xvY2FsZV9zdHJpbmdfcmVmLCB1c2VyX2xvY2FsZSwgc2l6ZW9mKHVzZXJfbG9jYWxlKSwga0NGU3RyaW5nRW5jb2RpbmdVVEY4ICk7CiAgICBDRlJlbGVhc2UoIHVzZXJfbG9jYWxlX3JlZiApOwogICAgaWYgKCFzdHJjaHIoIHVzZXJfbG9jYWxlLCAnLicgKSkgc3RyY2F0KCB1c2VyX2xvY2FsZSwgIi5VVEYtOCIgKTsKICAgIHVuaXhfY3AgPSBDUF9VVEY4OyAgLyogZGVmYXVsdCB0byB1dGYtOCBldmVuIGlmIHdlIGRvbid0IGdldCBhIHZhbGlkIGxvY2FsZSAqLwogICAgc2V0ZW52KCAiTEFORyIsIHVzZXJfbG9jYWxlLCAwICk7CiAgICBUUkFDRSggInNldHRpbmcgbG9jYWxlIHRvICclcydcbiIsIHVzZXJfbG9jYWxlICk7CgogICAgLyogV2Ugc3RpbGwgd2FudCB0byBzZXQgdGhlIHJldHJpZXZlIHRoZSBwcmVmZXJyZWQgbGFuZ3VhZ2UgYXMgY2hvc2VuIGluCiAgICAgICBTeXN0ZW0gUHJlZmVyZW5jZXMuYXBwLCBiZWNhdXNlIGl0IGNhbiBkaWZmZXIgZnJvbSBDRkxvY2FsZUNvcHlDdXJyZW50KCkuCiAgICAqLwogICAgYWxsX2xvY2FsZXMgPSBDRkxvY2FsZUNvcHlBdmFpbGFibGVMb2NhbGVJZGVudGlmaWVycygpOwogICAgcHJlZmVycmVkX2xvY2FsZXMgPSBDRkJ1bmRsZUNvcHlMb2NhbGl6YXRpb25zRm9yUHJlZmVyZW5jZXMoIGFsbF9sb2NhbGVzLCBOVUxMICk7CiAgICBpZiAocHJlZmVycmVkX2xvY2FsZXMgJiYgQ0ZBcnJheUdldENvdW50KCBwcmVmZXJyZWRfbG9jYWxlcyApKQogICAgICAgIHVzZXJfbGFuZ3VhZ2Vfc3RyaW5nX3JlZiA9IENGQXJyYXlHZXRWYWx1ZUF0SW5kZXgoIHByZWZlcnJlZF9sb2NhbGVzLCAwICk7CiAgICBDRlJlbGVhc2UoIGFsbF9sb2NhbGVzICk7CiNlbmRpZiAvKiBfX0FQUExFX18gKi8KCiAgICBzZXRsb2NhbGUoIExDX0FMTCwgIiIgKTsKCiAgICB1bml4X2NwID0gc2V0dXBfdW5peF9sb2NhbGVzKCk7CiAgICBpZiAoIWxjaWRfTENfTUVTU0FHRVMpIGxjaWRfTENfTUVTU0FHRVMgPSBsY2lkX0xDX0NUWVBFOwoKI2lmZGVmIF9fQVBQTEVfXwogICAgLyogT3ZlcnJpZGUgbGNpZF9MQ19NRVNTQUdFUyB3aXRoIHVzZXJfbGFuZ3VhZ2UgaWYgTENfTUVTU0FHRVMgaXMgc2V0IHRvIGRlZmF1bHQgKi8KICAgIGlmIChsY2lkX0xDX01FU1NBR0VTID09IGxjaWRfTENfQ1RZUEUgJiYgdXNlcl9sYW5ndWFnZV9zdHJpbmdfcmVmKQogICAgewogICAgICAgIHN0cnVjdCBsb2NhbGVfbmFtZSBsb2NhbGVfbmFtZTsKICAgICAgICBXQ0hBUiBidWZmZXJbMTI4XTsKICAgICAgICBDRlN0cmluZ0dldENTdHJpbmcoIHVzZXJfbGFuZ3VhZ2Vfc3RyaW5nX3JlZiwgdXNlcl9sb2NhbGUsIHNpemVvZih1c2VyX2xvY2FsZSksIGtDRlN0cmluZ0VuY29kaW5nVVRGOCApOwogICAgICAgIHN0cmNweW5BdG9XKCBidWZmZXIsIHVzZXJfbG9jYWxlLCBzaXplb2YoYnVmZmVyKS9zaXplb2YoV0NIQVIpICk7CiAgICAgICAgcGFyc2VfbG9jYWxlX25hbWUoIGJ1ZmZlciwgJmxvY2FsZV9uYW1lICk7CiAgICAgICAgbGNpZF9MQ19NRVNTQUdFUyA9IGxvY2FsZV9uYW1lLmxjaWQ7CiAgICAgICAgVFJBQ0UoICJzZXR0aW5nIGxjaWRfTENfTUVTU0FHRVMgdG8gJyVzJ1xuIiwgdXNlcl9sb2NhbGUgKTsKICAgIH0KICAgIGlmIChwcmVmZXJyZWRfbG9jYWxlcykKICAgICAgICBDRlJlbGVhc2UoIHByZWZlcnJlZF9sb2NhbGVzICk7CiNlbmRpZgoKICAgIE50U2V0RGVmYXVsdFVJTGFuZ3VhZ2UoIExBTkdJREZST01MQ0lEKGxjaWRfTENfTUVTU0FHRVMpICk7CiAgICBOdFNldERlZmF1bHRMb2NhbGUoIFRSVUUsIGxjaWRfTENfTUVTU0FHRVMgKTsKICAgIE50U2V0RGVmYXVsdExvY2FsZSggRkFMU0UsIGxjaWRfTENfQ1RZUEUgKTsKCiAgICBhbnNpX2NwID0gZ2V0X2xjaWRfY29kZXBhZ2UoIExPQ0FMRV9VU0VSX0RFRkFVTFQgKTsKICAgIEdldExvY2FsZUluZm9XKCBMT0NBTEVfVVNFUl9ERUZBVUxULCBMT0NBTEVfSURFRkFVTFRNQUNDT0RFUEFHRSB8IExPQ0FMRV9SRVRVUk5fTlVNQkVSLAogICAgICAgICAgICAgICAgICAgIChMUFdTVFIpJm1hY19jcCwgc2l6ZW9mKG1hY19jcCkvc2l6ZW9mKFdDSEFSKSApOwogICAgR2V0TG9jYWxlSW5mb1coIExPQ0FMRV9VU0VSX0RFRkFVTFQsIExPQ0FMRV9JREVGQVVMVENPREVQQUdFIHwgTE9DQUxFX1JFVFVSTl9OVU1CRVIsCiAgICAgICAgICAgICAgICAgICAgKExQV1NUUikmb2VtX2NwLCBzaXplb2Yob2VtX2NwKS9zaXplb2YoV0NIQVIpICk7CiAgICBpZiAoIXVuaXhfY3ApCiAgICAgICAgR2V0TG9jYWxlSW5mb1coIExPQ0FMRV9VU0VSX0RFRkFVTFQsIExPQ0FMRV9JREVGQVVMVFVOSVhDT0RFUEFHRSB8IExPQ0FMRV9SRVRVUk5fTlVNQkVSLAogICAgICAgICAgICAgICAgICAgICAgICAoTFBXU1RSKSZ1bml4X2NwLCBzaXplb2YodW5peF9jcCkvc2l6ZW9mKFdDSEFSKSApOwoKICAgIGlmICghKGFuc2lfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBhbnNpX2NwICkpKQogICAgICAgIGFuc2lfY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAxMjUyICk7CiAgICBpZiAoIShvZW1fY3B0YWJsZSA9IHdpbmVfY3BfZ2V0X3RhYmxlKCBvZW1fY3AgKSkpCiAgICAgICAgb2VtX2NwdGFibGUgID0gd2luZV9jcF9nZXRfdGFibGUoIDQzNyApOwogICAgaWYgKCEobWFjX2NwdGFibGUgPSB3aW5lX2NwX2dldF90YWJsZSggbWFjX2NwICkpKQogICAgICAgIG1hY19jcHRhYmxlICA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAxMDAwMCApOwogICAgaWYgKHVuaXhfY3AgIT0gQ1BfVVRGOCkKICAgIHsKICAgICAgICBpZiAoISh1bml4X2NwdGFibGUgPSB3aW5lX2NwX2dldF90YWJsZSggdW5peF9jcCApKSkKICAgICAgICAgICAgdW5peF9jcHRhYmxlICA9IHdpbmVfY3BfZ2V0X3RhYmxlKCAyODU5MSApOwogICAgfQoKICAgIF9fd2luZV9pbml0X2NvZGVwYWdlcyggYW5zaV9jcHRhYmxlLCBvZW1fY3B0YWJsZSwgdW5peF9jcHRhYmxlICk7CgogICAgVFJBQ0UoICJhbnNpPSUwM2Qgb2VtPSUwM2QgbWFjPSUwM2QgdW5peD0lMDNkXG4iLAogICAgICAgICAgIGFuc2lfY3B0YWJsZS0+aW5mby5jb2RlcGFnZSwgb2VtX2NwdGFibGUtPmluZm8uY29kZXBhZ2UsCiAgICAgICAgICAgbWFjX2NwdGFibGUtPmluZm8uY29kZXBhZ2UsIHVuaXhfY3AgKTsKCiAgICBzZXRsb2NhbGUoTENfTlVNRVJJQywgIkMiKTsgIC8qIEZJWE1FOiBvbGVhdXQzMiBkZXBlbmRzIG9uIHRoaXMgKi8KfQoKc3RhdGljIEhBTkRMRSBOTFNfUmVnT3BlbktleShIQU5ETEUgaFJvb3RLZXksIExQQ1dTVFIgc3pLZXlOYW1lKQp7CiAgICBVTklDT0RFX1NUUklORyBrZXlOYW1lOwogICAgT0JKRUNUX0FUVFJJQlVURVMgYXR0cjsKICAgIEhBTkRMRSBoa2V5OwoKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAma2V5TmFtZSwgc3pLZXlOYW1lICk7CiAgICBJbml0aWFsaXplT2JqZWN0QXR0cmlidXRlcygmYXR0ciwgJmtleU5hbWUsIDAsIGhSb290S2V5LCBOVUxMKTsKCiAgICBpZiAoTnRPcGVuS2V5KCAmaGtleSwgS0VZX0FMTF9BQ0NFU1MsICZhdHRyICkgIT0gU1RBVFVTX1NVQ0NFU1MpCiAgICAgICAgaGtleSA9IDA7CgogICAgcmV0dXJuIGhrZXk7Cn0KCnN0YXRpYyBCT09MIE5MU19SZWdFbnVtU3ViS2V5KEhBTkRMRSBoS2V5LCBVSU5UIHVsSW5kZXgsIExQV1NUUiBzektleU5hbWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVMT05HIGtleU5hbWVTaXplKQp7CiAgICBCWVRFIGJ1ZmZlcls4MF07CiAgICBLRVlfQkFTSUNfSU5GT1JNQVRJT04gKmluZm8gPSAoS0VZX0JBU0lDX0lORk9STUFUSU9OICopYnVmZmVyOwogICAgRFdPUkQgZHdMZW47CgogICAgaWYgKE50RW51bWVyYXRlS2V5KCBoS2V5LCB1bEluZGV4LCBLZXlCYXNpY0luZm9ybWF0aW9uLCBidWZmZXIsCiAgICAgICAgICAgICAgICAgICAgICAgIHNpemVvZihidWZmZXIpLCAmZHdMZW4pICE9IFNUQVRVU19TVUNDRVNTIHx8CiAgICAgICAgaW5mby0+TmFtZUxlbmd0aCA+IGtleU5hbWVTaXplKQogICAgewogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBUUkFDRSgiaW5mby0+TmFtZSAlcyBpbmZvLT5OYW1lTGVuZ3RoICVkXG4iLCBkZWJ1Z3N0cl93KGluZm8tPk5hbWUpLCBpbmZvLT5OYW1lTGVuZ3RoKTsKCiAgICBtZW1jcHkoIHN6S2V5TmFtZSwgaW5mby0+TmFtZSwgaW5mby0+TmFtZUxlbmd0aCk7CiAgICBzektleU5hbWVbaW5mby0+TmFtZUxlbmd0aCAvIHNpemVvZihXQ0hBUildID0gJ1wwJzsKCiAgICBUUkFDRSgicmV0dXJuaW5nICVzXG4iLCBkZWJ1Z3N0cl93KHN6S2V5TmFtZSkpOwogICAgcmV0dXJuIFRSVUU7Cn0KCnN0YXRpYyBCT09MIE5MU19SZWdFbnVtVmFsdWUoSEFORExFIGhLZXksIFVJTlQgdWxJbmRleCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUFdTVFIgc3pWYWx1ZU5hbWUsIFVMT05HIHZhbHVlTmFtZVNpemUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTFBXU1RSIHN6VmFsdWVEYXRhLCBVTE9ORyB2YWx1ZURhdGFTaXplKQp7CiAgICBCWVRFIGJ1ZmZlcls4MF07CiAgICBLRVlfVkFMVUVfRlVMTF9JTkZPUk1BVElPTiAqaW5mbyA9IChLRVlfVkFMVUVfRlVMTF9JTkZPUk1BVElPTiAqKWJ1ZmZlcjsKICAgIERXT1JEIGR3TGVuOwoKICAgIGlmIChOdEVudW1lcmF0ZVZhbHVlS2V5KCBoS2V5LCB1bEluZGV4LCBLZXlWYWx1ZUZ1bGxJbmZvcm1hdGlvbiwKICAgICAgICBidWZmZXIsIHNpemVvZihidWZmZXIpLCAmZHdMZW4gKSAhPSBTVEFUVVNfU1VDQ0VTUyB8fAogICAgICAgIGluZm8tPk5hbWVMZW5ndGggPiB2YWx1ZU5hbWVTaXplIHx8CiAgICAgICAgaW5mby0+RGF0YUxlbmd0aCA+IHZhbHVlRGF0YVNpemUpCiAgICB7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIFRSQUNFKCJpbmZvLT5OYW1lICVzIGluZm8tPkRhdGFMZW5ndGggJWRcbiIsIGRlYnVnc3RyX3coaW5mby0+TmFtZSksIGluZm8tPkRhdGFMZW5ndGgpOwoKICAgIG1lbWNweSggc3pWYWx1ZU5hbWUsIGluZm8tPk5hbWUsIGluZm8tPk5hbWVMZW5ndGgpOwogICAgc3pWYWx1ZU5hbWVbaW5mby0+TmFtZUxlbmd0aCAvIHNpemVvZihXQ0hBUildID0gJ1wwJzsKICAgIG1lbWNweSggc3pWYWx1ZURhdGEsIGJ1ZmZlciArIGluZm8tPkRhdGFPZmZzZXQsIGluZm8tPkRhdGFMZW5ndGggKTsKICAgIHN6VmFsdWVEYXRhW2luZm8tPkRhdGFMZW5ndGggLyBzaXplb2YoV0NIQVIpXSA9ICdcMCc7CgogICAgVFJBQ0UoInJldHVybmluZyAlcyAlc1xuIiwgZGVidWdzdHJfdyhzelZhbHVlTmFtZSksIGRlYnVnc3RyX3coc3pWYWx1ZURhdGEpKTsKICAgIHJldHVybiBUUlVFOwp9CgpzdGF0aWMgQk9PTCBOTFNfUmVnR2V0RHdvcmQoSEFORExFIGhLZXksIExQQ1dTVFIgc3pWYWx1ZU5hbWUsIERXT1JEICpscFZhbCkKewogICAgQllURSBidWZmZXJbMTI4XTsKICAgIGNvbnN0IEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICppbmZvID0gKEtFWV9WQUxVRV9QQVJUSUFMX0lORk9STUFUSU9OICopYnVmZmVyOwogICAgRFdPUkQgZHdTaXplID0gc2l6ZW9mKGJ1ZmZlcik7CiAgICBVTklDT0RFX1NUUklORyB2YWx1ZU5hbWU7CgogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZ2YWx1ZU5hbWUsIHN6VmFsdWVOYW1lICk7CgogICAgVFJBQ0UoIiVwLCAlc1xuIiwgaEtleSwgZGVidWdzdHJfdyhzelZhbHVlTmFtZSkpOwogICAgaWYgKE50UXVlcnlWYWx1ZUtleSggaEtleSwgJnZhbHVlTmFtZSwgS2V5VmFsdWVQYXJ0aWFsSW5mb3JtYXRpb24sCiAgICAgICAgICAgICAgICAgICAgICAgICBidWZmZXIsIGR3U2l6ZSwgJmR3U2l6ZSApID09IFNUQVRVU19TVUNDRVNTICYmCiAgICAgICAgaW5mby0+RGF0YUxlbmd0aCA9PSBzaXplb2YoRFdPUkQpKQogICAgewogICAgICAgIG1lbWNweShscFZhbCwgaW5mby0+RGF0YSwgc2l6ZW9mKERXT1JEKSk7CiAgICAgICAgcmV0dXJuIFRSVUU7CiAgICB9CgogICAgcmV0dXJuIEZBTFNFOwp9CgpzdGF0aWMgQk9PTCBOTFNfR2V0TGFuZ3VhZ2VHcm91cE5hbWUoTEdSUElEIGxncnBpZCwgTFBXU1RSIHN6TmFtZSwgVUxPTkcgbmFtZVNpemUpCnsKICAgIExBTkdJRCAgbGFuZ0lkOwogICAgTFBDV1NUUiBzelJlc291cmNlTmFtZSA9IE1BS0VJTlRSRVNPVVJDRVcoKChsZ3JwaWQgKyAweDIwMDApID4+IDQpICsgMSk7CiAgICBIUlNSQyAgIGhSZXNvdXJjZTsKICAgIEJPT0wgICAgYlJldCA9IEZBTFNFOwoKICAgIC8qIEZJWE1FOiBJcyBpdCBjb3JyZWN0IHRvIHVzZSB0aGUgc3lzdGVtIGRlZmF1bHQgbGFuZ2lkPyAqLwogICAgbGFuZ0lkID0gR2V0U3lzdGVtRGVmYXVsdExhbmdJRCgpOwoKICAgIGlmIChTVUJMQU5HSUQobGFuZ0lkKSA9PSBTVUJMQU5HX05FVVRSQUwpCiAgICAgICAgbGFuZ0lkID0gTUFLRUxBTkdJRCggUFJJTUFSWUxBTkdJRChsYW5nSWQpLCBTVUJMQU5HX0RFRkFVTFQgKTsKCiAgICBoUmVzb3VyY2UgPSBGaW5kUmVzb3VyY2VFeFcoIGtlcm5lbDMyX2hhbmRsZSwgKExQV1NUUilSVF9TVFJJTkcsIHN6UmVzb3VyY2VOYW1lLCBsYW5nSWQgKTsKCiAgICBpZiAoaFJlc291cmNlKQogICAgewogICAgICAgIEhHTE9CQUwgaFJlc0RpciA9IExvYWRSZXNvdXJjZSgga2VybmVsMzJfaGFuZGxlLCBoUmVzb3VyY2UgKTsKCiAgICAgICAgaWYgKGhSZXNEaXIpCiAgICAgICAgewogICAgICAgICAgICBVTE9ORyAgIGlSZXNvdXJjZUluZGV4ID0gbGdycGlkICYgMHhmOwogICAgICAgICAgICBMUENXU1RSIGxwUmVzRW50cnkgPSBMb2NrUmVzb3VyY2UoIGhSZXNEaXIgKTsKICAgICAgICAgICAgVUxPTkcgICBpOwoKICAgICAgICAgICAgZm9yIChpID0gMDsgaSA8IGlSZXNvdXJjZUluZGV4OyBpKyspCiAgICAgICAgICAgICAgICBscFJlc0VudHJ5ICs9ICpscFJlc0VudHJ5ICsgMTsKCiAgICAgICAgICAgIGlmICgqbHBSZXNFbnRyeSA8IG5hbWVTaXplKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBtZW1jcHkoIHN6TmFtZSwgbHBSZXNFbnRyeSArIDEsICpscFJlc0VudHJ5ICogc2l6ZW9mKFdDSEFSKSApOwogICAgICAgICAgICAgICAgc3pOYW1lWypscFJlc0VudHJ5XSA9ICdcMCc7CiAgICAgICAgICAgICAgICBiUmV0ID0gVFJVRTsKICAgICAgICAgICAgfQoKICAgICAgICB9CiAgICAgICAgRnJlZVJlc291cmNlKCBoUmVzb3VyY2UgKTsKICAgIH0KICAgIHJldHVybiBiUmV0Owp9CgovKiBSZWdpc3RyeSBrZXlzIGZvciBOTFMgcmVsYXRlZCBpbmZvcm1hdGlvbiAqLwoKc3RhdGljIGNvbnN0IFdDSEFSIHN6Q291bnRyeUxpc3ROYW1lW10gPSB7CiAgICAnTScsJ2EnLCdjJywnaCcsJ2knLCduJywnZScsJ1xcJywnUycsJ28nLCdmJywndCcsJ3cnLCdhJywncicsJ2UnLCdcXCcsCiAgICAnTScsJ2knLCdjJywncicsJ28nLCdzJywnbycsJ2YnLCd0JywnXFwnLCdXJywnaScsJ24nLCdkJywnbycsJ3cnLCdzJywnXFwnLAogICAgJ0MnLCd1JywncicsJ3InLCdlJywnbicsJ3QnLCdWJywnZScsJ3InLCdzJywnaScsJ28nLCduJywnXFwnLAogICAgJ1QnLCdlJywnbCcsJ2UnLCdwJywnaCcsJ28nLCduJywneScsJ1xcJywKICAgICdDJywnbycsJ3UnLCduJywndCcsJ3InLCd5JywnICcsJ0wnLCdpJywncycsJ3QnLCdcMCcKfTsKCgovKiBDYWxsYmFjayBmdW5jdGlvbiBwdHJzIGZvciBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBL1cgKi8KdHlwZWRlZiBzdHJ1Y3QKewogIExBTkdVQUdFR1JPVVBfRU5VTVBST0NBIHByb2NBOwogIExBTkdVQUdFR1JPVVBfRU5VTVBST0NXIHByb2NXOwogIERXT1JEICAgIGR3RmxhZ3M7CiAgTE9OR19QVFIgbFBhcmFtOwp9IEVOVU1MQU5HVUFHRUdST1VQX0NBTExCQUNLUzsKCi8qIEludGVybmFsIGltcGxlbWVudGF0aW9uIG9mIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc0EvVyAqLwpzdGF0aWMgQk9PTCBOTFNfRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzKEVOVU1MQU5HVUFHRUdST1VQX0NBTExCQUNLUyAqbHBQcm9jcykKewogICAgV0NIQVIgc3pOdW1iZXJbMTBdLCBzelZhbHVlWzRdOwogICAgSEFORExFIGhLZXk7CiAgICBCT09MIGJDb250aW51ZSA9IFRSVUU7CiAgICBVTE9ORyB1bEluZGV4ID0gMDsKCiAgICBpZiAoIWxwUHJvY3MpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgc3dpdGNoIChscFByb2NzLT5kd0ZsYWdzKQogICAgewogICAgY2FzZSAwOgogICAgICAgIC8qIERlZmF1bHQgdG8gTEdSUElEX0lOU1RBTExFRCAqLwogICAgICAgIGxwUHJvY3MtPmR3RmxhZ3MgPSBMR1JQSURfSU5TVEFMTEVEOwogICAgICAgIC8qIEZhbGwgdGhyb3VnaC4uLiAqLwogICAgY2FzZSBMR1JQSURfSU5TVEFMTEVEOgogICAgY2FzZSBMR1JQSURfU1VQUE9SVEVEOgogICAgICAgIGJyZWFrOwogICAgZGVmYXVsdDoKICAgICAgICBTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGhLZXkgPSBOTFNfUmVnT3BlbktleSggMCwgc3pMYW5nR3JvdXBzS2V5TmFtZSApOwoKICAgIGlmICghaEtleSkKICAgICAgICBGSVhNRSgiTkxTIHJlZ2lzdHJ5IGtleSBub3QgZm91bmQuIFBsZWFzZSBhcHBseSB0aGUgZGVmYXVsdCByZWdpc3RyeSBmaWxlICd3aW5lLmluZidcbiIpOwoKICAgIHdoaWxlIChiQ29udGludWUpCiAgICB7CiAgICAgICAgaWYgKE5MU19SZWdFbnVtVmFsdWUoIGhLZXksIHVsSW5kZXgsIHN6TnVtYmVyLCBzaXplb2Yoc3pOdW1iZXIpLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzelZhbHVlLCBzaXplb2Yoc3pWYWx1ZSkgKSkKICAgICAgICB7CiAgICAgICAgICAgIEJPT0wgYkluc3RhbGxlZCA9IHN6VmFsdWVbMF0gPT0gJzEnID8gVFJVRSA6IEZBTFNFOwogICAgICAgICAgICBMR1JQSUQgbGdycGlkID0gc3RydG91bFcoIHN6TnVtYmVyLCBOVUxMLCAxNiApOwoKICAgICAgICAgICAgVFJBQ0UoImdycGlkICVzICglc2luc3RhbGxlZClcbiIsIGRlYnVnc3RyX3coc3pOdW1iZXIpLAogICAgICAgICAgICAgICAgICAgYkluc3RhbGxlZCA/ICIiIDogIm5vdCAiKTsKCiAgICAgICAgICAgIGlmIChscFByb2NzLT5kd0ZsYWdzID09IExHUlBJRF9TVVBQT1JURUQgfHwgYkluc3RhbGxlZCkKICAgICAgICAgICAgewogICAgICAgICAgICAgICAgV0NIQVIgc3pHcnBOYW1lWzQ4XTsKCiAgICAgICAgICAgICAgICBpZiAoIU5MU19HZXRMYW5ndWFnZUdyb3VwTmFtZSggbGdycGlkLCBzekdycE5hbWUsIHNpemVvZihzekdycE5hbWUpIC8gc2l6ZW9mKFdDSEFSKSApKQogICAgICAgICAgICAgICAgICAgIHN6R3JwTmFtZVswXSA9ICdcMCc7CgogICAgICAgICAgICAgICAgaWYgKGxwUHJvY3MtPnByb2NXKQogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NXKCBsZ3JwaWQsIHN6TnVtYmVyLCBzekdycE5hbWUsIGxwUHJvY3MtPmR3RmxhZ3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxwUHJvY3MtPmxQYXJhbSApOwogICAgICAgICAgICAgICAgZWxzZQogICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgIGNoYXIgc3pOdW1iZXJBW3NpemVvZihzek51bWJlcikvc2l6ZW9mKFdDSEFSKV07CiAgICAgICAgICAgICAgICAgICAgY2hhciBzekdycE5hbWVBWzQ4XTsKCiAgICAgICAgICAgICAgICAgICAgLyogRklYTUU6IE1TRE4gZG9lc24ndCBzYXkgd2hpY2ggY29kZSBwYWdlIHRoZSBXLT5BIHRyYW5zbGF0aW9uIHVzZXMsCiAgICAgICAgICAgICAgICAgICAgICogICAgICAgIG9yIHdoZXRoZXIgdGhlIGxhbmd1YWdlIG5hbWVzIGFyZSBldmVyIGxvY2FsaXNlZC4gQXNzdW1lIENQX0FDUC4KICAgICAgICAgICAgICAgICAgICAgKi8KCiAgICAgICAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIHN6TnVtYmVyLCAtMSwgc3pOdW1iZXJBLCBzaXplb2Yoc3pOdW1iZXJBKSwgMCwgMCk7CiAgICAgICAgICAgICAgICAgICAgV2lkZUNoYXJUb011bHRpQnl0ZShDUF9BQ1AsIDAsIHN6R3JwTmFtZSwgLTEsIHN6R3JwTmFtZUEsIHNpemVvZihzekdycE5hbWVBKSwgMCwgMCk7CgogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NBKCBsZ3JwaWQsIHN6TnVtYmVyQSwgc3pHcnBOYW1lQSwgbHBQcm9jcy0+ZHdGbGFncywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgbHBQcm9jcy0+bFBhcmFtICk7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KCiAgICAgICAgICAgIHVsSW5kZXgrKzsKICAgICAgICB9CiAgICAgICAgZWxzZQogICAgICAgICAgICBiQ29udGludWUgPSBGQUxTRTsKCiAgICAgICAgaWYgKCFiQ29udGludWUpCiAgICAgICAgICAgIGJyZWFrOwogICAgfQoKICAgIGlmIChoS2V5KQogICAgICAgIE50Q2xvc2UoIGhLZXkgKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzQSAgICAoS0VSTkVMMzIuQCkKICoKICogQ2FsbCBhIHVzZXJzIGZ1bmN0aW9uIGZvciBlYWNoIGxhbmd1YWdlIGdyb3VwIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIHBMYW5nR3JwRW51bVByb2MgW0ldIENhbGxiYWNrIGZ1bmN0aW9uIHRvIGNhbGwgZm9yIGVhY2ggbGFuZ3VhZ2UgZ3JvdXAKICogIGR3RmxhZ3MgICAgICAgICAgW0ldIExHUlBJRF9TVVBQT1JURUQ9QWxsIFN1cHBvcnRlZCwgTEdSUElEX0lOU1RBTExFRD1JbnN0YWxsZWQgb25seQogKiAgbFBhcmFtICAgICAgICAgICBbSV0gVXNlciBwYXJhbWV0ZXIgdG8gcGFzcyB0byBwTGFuZ0dycEVudW1Qcm9jCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNBKExBTkdVQUdFR1JPVVBfRU5VTVBST0NBIHBMYW5nR3JwRW51bVByb2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNTEFOR1VBR0VHUk9VUF9DQUxMQkFDS1MgcHJvY3M7CgogICAgVFJBQ0UoIiglcCwweCUwOFgsMHglMDhsWClcbiIsIHBMYW5nR3JwRW51bVByb2MsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgcHJvY3MucHJvY0EgPSBwTGFuZ0dycEVudW1Qcm9jOwogICAgcHJvY3MucHJvY1cgPSBOVUxMOwogICAgcHJvY3MuZHdGbGFncyA9IGR3RmxhZ3M7CiAgICBwcm9jcy5sUGFyYW0gPSBsUGFyYW07CgogICAgcmV0dXJuIE5MU19FbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHMoIHBMYW5nR3JwRW51bVByb2MgPyAmcHJvY3MgOiBOVUxMKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUxhbmd1YWdlR3JvdXBzVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEVudW1TeXN0ZW1MYW5ndWFnZUdyb3Vwc0EuCiAqLwpCT09MIFdJTkFQSSBFbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHNXKExBTkdVQUdFR1JPVVBfRU5VTVBST0NXIHBMYW5nR3JwRW51bVByb2MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNTEFOR1VBR0VHUk9VUF9DQUxMQkFDS1MgcHJvY3M7CgogICAgVFJBQ0UoIiglcCwweCUwOFgsMHglMDhsWClcbiIsIHBMYW5nR3JwRW51bVByb2MsIGR3RmxhZ3MsIGxQYXJhbSk7CgogICAgcHJvY3MucHJvY0EgPSBOVUxMOwogICAgcHJvY3MucHJvY1cgPSBwTGFuZ0dycEVudW1Qcm9jOwogICAgcHJvY3MuZHdGbGFncyA9IGR3RmxhZ3M7CiAgICBwcm9jcy5sUGFyYW0gPSBsUGFyYW07CgogICAgcmV0dXJuIE5MU19FbnVtU3lzdGVtTGFuZ3VhZ2VHcm91cHMoIHBMYW5nR3JwRW51bVByb2MgPyAmcHJvY3MgOiBOVUxMKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgSXNWYWxpZExhbmd1YWdlR3JvdXAgICAgKEtFUk5FTDMyLkApCiAqCiAqIERldGVybWluZSBpZiBhIGxhbmd1YWdlIGdyb3VwIGlzIHN1cHBvcnRlZCBhbmQvb3IgaW5zdGFsbGVkLgogKgogKiBQQVJBTVMKICogIGxncnBpZCAgW0ldIExhbmd1YWdlIEdyb3VwIElkIChMR1JQSURfIHZhbHVlcyBmcm9tICJ3aW5ubHMuaCIpCiAqICBkd0ZsYWdzIFtJXSBMR1JQSURfU1VQUE9SVEVEPVN1cHBvcnRlZCwgTEdSUElEX0lOU1RBTExFRD1JbnN0YWxsZWQKICoKICogUkVUVVJOUwogKiAgVFJVRSwgaWYgbGdycGlkIGlzIHN1cHBvcnRlZCBhbmQvb3IgaW5zdGFsbGVkLCBhY2NvcmRpbmcgdG8gZHdGbGFncy4KICogIEZBTFNFIG90aGVyd2lzZS4KICovCkJPT0wgV0lOQVBJIElzVmFsaWRMYW5ndWFnZUdyb3VwKExHUlBJRCBsZ3JwaWQsIERXT1JEIGR3RmxhZ3MpCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzekZvcm1hdFtdID0geyAnJScsJ3gnLCdcMCcgfTsKICAgIFdDSEFSIHN6VmFsdWVOYW1lWzE2XSwgc3pWYWx1ZVsyXTsKICAgIEJPT0wgYlN1cHBvcnRlZCA9IEZBTFNFLCBiSW5zdGFsbGVkID0gRkFMU0U7CiAgICBIQU5ETEUgaEtleTsKCgogICAgc3dpdGNoIChkd0ZsYWdzKQogICAgewogICAgY2FzZSBMR1JQSURfSU5TVEFMTEVEOgogICAgY2FzZSBMR1JQSURfU1VQUE9SVEVEOgoKICAgICAgICBoS2V5ID0gTkxTX1JlZ09wZW5LZXkoIDAsIHN6TGFuZ0dyb3Vwc0tleU5hbWUgKTsKCiAgICAgICAgc3ByaW50ZlcoIHN6VmFsdWVOYW1lLCBzekZvcm1hdCwgbGdycGlkICk7CgogICAgICAgIGlmIChOTFNfUmVnR2V0RHdvcmQoIGhLZXksIHN6VmFsdWVOYW1lLCAoTFBEV09SRCkmc3pWYWx1ZSApKQogICAgICAgIHsKICAgICAgICAgICAgYlN1cHBvcnRlZCA9IFRSVUU7CgogICAgICAgICAgICBpZiAoc3pWYWx1ZVswXSA9PSAnMScpCiAgICAgICAgICAgICAgICBiSW5zdGFsbGVkID0gVFJVRTsKICAgICAgICB9CgogICAgICAgIGlmIChoS2V5KQogICAgICAgICAgICBOdENsb3NlKCBoS2V5ICk7CgogICAgICAgIGJyZWFrOwogICAgfQoKICAgIGlmICgoZHdGbGFncyA9PSBMR1JQSURfU1VQUE9SVEVEICYmIGJTdXBwb3J0ZWQpIHx8CiAgICAgICAgKGR3RmxhZ3MgPT0gTEdSUElEX0lOU1RBTExFRCAmJiBiSW5zdGFsbGVkKSkKICAgICAgICByZXR1cm4gVFJVRTsKCiAgICByZXR1cm4gRkFMU0U7Cn0KCi8qIENhbGxiYWNrIGZ1bmN0aW9uIHB0cnMgZm9yIEVudW1MYW5ndWFnZUdyb3VwbG9jYWxlc0EvVyAqLwp0eXBlZGVmIHN0cnVjdAp7CiAgTEFOR0dST1VQTE9DQUxFX0VOVU1QUk9DQSBwcm9jQTsKICBMQU5HR1JPVVBMT0NBTEVfRU5VTVBST0NXIHByb2NXOwogIERXT1JEICAgIGR3RmxhZ3M7CiAgTEdSUElEICAgbGdycGlkOwogIExPTkdfUFRSIGxQYXJhbTsKfSBFTlVNTEFOR1VBR0VHUk9VUExPQ0FMRV9DQUxMQkFDS1M7CgovKiBJbnRlcm5hbCBpbXBsZW1lbnRhdGlvbiBvZiBFbnVtTGFuZ3VhZ2VHcm91cGxvY2FsZXNBL1cgKi8Kc3RhdGljIEJPT0wgTkxTX0VudW1MYW5ndWFnZUdyb3VwTG9jYWxlcyhFTlVNTEFOR1VBR0VHUk9VUExPQ0FMRV9DQUxMQkFDS1MgKmxwUHJvY3MpCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBzekFsdGVybmF0ZVNvcnRzS2V5TmFtZVtdID0gewogICAgICAnQScsJ2wnLCd0JywnZScsJ3InLCduJywnYScsJ3QnLCdlJywnICcsJ1MnLCdvJywncicsJ3QnLCdzJywnXDAnCiAgICB9OwogICAgV0NIQVIgc3pOdW1iZXJbMTBdLCBzelZhbHVlWzRdOwogICAgSEFORExFIGhLZXk7CiAgICBCT09MIGJDb250aW51ZSA9IFRSVUUsIGJBbHRlcm5hdGUgPSBGQUxTRTsKICAgIExHUlBJRCBsZ3JwaWQ7CiAgICBVTE9ORyB1bEluZGV4ID0gMTsgIC8qIElnbm9yZSBkZWZhdWx0IGVudHJ5IG9mIDFzdCBrZXkgKi8KCiAgICBpZiAoIWxwUHJvY3MgfHwgIWxwUHJvY3MtPmxncnBpZCB8fCBscFByb2NzLT5sZ3JwaWQgPiBMR1JQSURfQVJNRU5JQU4pCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKICAgICAgICByZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaWYgKGxwUHJvY3MtPmR3RmxhZ3MpCiAgICB7CiAgICAgICAgU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfRkxBR1MpOwogICAgICAgIHJldHVybiBGQUxTRTsKICAgIH0KCiAgICBoS2V5ID0gTkxTX1JlZ09wZW5LZXkoIDAsIHN6TG9jYWxlS2V5TmFtZSApOwoKICAgIGlmICghaEtleSkKICAgICAgICBXQVJOKCJOTFMgcmVnaXN0cnkga2V5IG5vdCBmb3VuZC4gUGxlYXNlIGFwcGx5IHRoZSBkZWZhdWx0IHJlZ2lzdHJ5IGZpbGUgJ3dpbmUuaW5mJ1xuIik7CgogICAgd2hpbGUgKGJDb250aW51ZSkKICAgIHsKICAgICAgICBpZiAoTkxTX1JlZ0VudW1WYWx1ZSggaEtleSwgdWxJbmRleCwgc3pOdW1iZXIsIHNpemVvZihzek51bWJlciksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN6VmFsdWUsIHNpemVvZihzelZhbHVlKSApKQogICAgICAgIHsKICAgICAgICAgICAgbGdycGlkID0gc3RydG91bFcoIHN6VmFsdWUsIE5VTEwsIDE2ICk7CgogICAgICAgICAgICBUUkFDRSgibGNpZCAlcywgZ3JwaWQgJWQgKCVzbWF0Y2hlZClcbiIsIGRlYnVnc3RyX3coc3pOdW1iZXIpLAogICAgICAgICAgICAgICAgICAgbGdycGlkLCBsZ3JwaWQgPT0gbHBQcm9jcy0+bGdycGlkID8gIiIgOiAibm90ICIpOwoKICAgICAgICAgICAgaWYgKGxncnBpZCA9PSBscFByb2NzLT5sZ3JwaWQpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIExDSUQgbGNpZDsKCiAgICAgICAgICAgICAgICBsY2lkID0gc3RydG91bFcoIHN6TnVtYmVyLCBOVUxMLCAxNiApOwoKICAgICAgICAgICAgICAgIC8qIEZJWE1FOiBuYXRpdmUgcmV0dXJucyBleHRyYSB0ZXh0IGZvciBhIGZldyAoMTcvMTUwKSBsb2NhbGVzLCBlLmc6CiAgICAgICAgICAgICAgICAgKiAnMDAwMDA0MzcgICAgICAgICAgO0dlb3JnaWFuJwogICAgICAgICAgICAgICAgICogQXQgcHJlc2VudCB3ZSBvbmx5IHBhc3MgdGhlIExDSUQgc3RyaW5nLgogICAgICAgICAgICAgICAgICovCgogICAgICAgICAgICAgICAgaWYgKGxwUHJvY3MtPnByb2NXKQogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IGxwUHJvY3MtPnByb2NXKCBsZ3JwaWQsIGxjaWQsIHN6TnVtYmVyLCBscFByb2NzLT5sUGFyYW0gKTsKICAgICAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgICAgICBjaGFyIHN6TnVtYmVyQVtzaXplb2Yoc3pOdW1iZXIpL3NpemVvZihXQ0hBUildOwoKICAgICAgICAgICAgICAgICAgICBXaWRlQ2hhclRvTXVsdGlCeXRlKENQX0FDUCwgMCwgc3pOdW1iZXIsIC0xLCBzek51bWJlckEsIHNpemVvZihzek51bWJlckEpLCAwLCAwKTsKCiAgICAgICAgICAgICAgICAgICAgYkNvbnRpbnVlID0gbHBQcm9jcy0+cHJvY0EoIGxncnBpZCwgbGNpZCwgc3pOdW1iZXJBLCBscFByb2NzLT5sUGFyYW0gKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQoKICAgICAgICAgICAgdWxJbmRleCsrOwogICAgICAgIH0KICAgICAgICBlbHNlCiAgICAgICAgewogICAgICAgICAgICAvKiBGaW5pc2hlZCBlbnVtZXJhdGluZyB0aGlzIGtleSAqLwogICAgICAgICAgICBpZiAoIWJBbHRlcm5hdGUpCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIC8qIEVudW1lcmF0ZSBhbHRlcm5hdGUgc29ydHMgYWxzbyAqLwogICAgICAgICAgICAgICAgaEtleSA9IE5MU19SZWdPcGVuS2V5KCBoS2V5LCBzekFsdGVybmF0ZVNvcnRzS2V5TmFtZSApOwogICAgICAgICAgICAgICAgYkFsdGVybmF0ZSA9IFRSVUU7CiAgICAgICAgICAgICAgICB1bEluZGV4ID0gMDsKICAgICAgICAgICAgfQogICAgICAgICAgICBlbHNlCiAgICAgICAgICAgICAgICBiQ29udGludWUgPSBGQUxTRTsgLyogRmluaXNoZWQgYm90aCBrZXlzICovCiAgICAgICAgfQoKICAgICAgICBpZiAoIWJDb250aW51ZSkKICAgICAgICAgICAgYnJlYWs7CiAgICB9CgogICAgaWYgKGhLZXkpCiAgICAgICAgTnRDbG9zZSggaEtleSApOwoKICAgIHJldHVybiBUUlVFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBFbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXNBICAgIChLRVJORUwzMi5AKQogKgogKiBDYWxsIGEgdXNlcnMgZnVuY3Rpb24gZm9yIGV2ZXJ5IGxvY2FsZSBpbiBhIGxhbmd1YWdlIGdyb3VwIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIHBMYW5nR3JwTGNFbnVtUHJvYyBbSV0gQ2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCBmb3IgZWFjaCBsb2NhbGUKICogIGxncnBpZCAgICAgICAgICAgICBbSV0gTGFuZ3VhZ2UgZ3JvdXAgKExHUlBJRF8gdmFsdWVzIGZyb20gIndpbm5scy5oIikKICogIGR3RmxhZ3MgICAgICAgICAgICBbSV0gUmVzZXJ2ZWQsIHNldCB0byAwCiAqICBsUGFyYW0gICAgICAgICAgICAgW0ldIFVzZXIgcGFyYW1ldGVyIHRvIHBhc3MgdG8gcExhbmdHcnBMY0VudW1Qcm9jCiAqCiAqIFJFVFVSTlMKICogIFN1Y2Nlc3M6IFRSVUUuCiAqICBGYWlsdXJlOiBGQUxTRS4gVXNlIEdldExhc3RFcnJvcigpIHRvIGRldGVybWluZSB0aGUgY2F1c2UuCiAqLwpCT09MIFdJTkFQSSBFbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXNBKExBTkdHUk9VUExPQ0FMRV9FTlVNUFJPQ0EgcExhbmdHcnBMY0VudW1Qcm9jLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExHUlBJRCBsZ3JwaWQsIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTUxBTkdVQUdFR1JPVVBMT0NBTEVfQ0FMTEJBQ0tTIGNhbGxiYWNrczsKCiAgICBUUkFDRSgiKCVwLDB4JTA4WCwweCUwOFgsMHglMDhsWClcbiIsIHBMYW5nR3JwTGNFbnVtUHJvYywgbGdycGlkLCBkd0ZsYWdzLCBsUGFyYW0pOwoKICAgIGNhbGxiYWNrcy5wcm9jQSAgID0gcExhbmdHcnBMY0VudW1Qcm9jOwogICAgY2FsbGJhY2tzLnByb2NXICAgPSBOVUxMOwogICAgY2FsbGJhY2tzLmR3RmxhZ3MgPSBkd0ZsYWdzOwogICAgY2FsbGJhY2tzLmxncnBpZCAgPSBsZ3JwaWQ7CiAgICBjYWxsYmFja3MubFBhcmFtICA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1MYW5ndWFnZUdyb3VwTG9jYWxlcyggcExhbmdHcnBMY0VudW1Qcm9jID8gJmNhbGxiYWNrcyA6IE5VTEwgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bUxhbmd1YWdlR3JvdXBMb2NhbGVzVyAgICAoS0VSTkVMMzIuQCkKICoKICogU2VlIEVudW1MYW5ndWFnZUdyb3VwTG9jYWxlc0EuCiAqLwpCT09MIFdJTkFQSSBFbnVtTGFuZ3VhZ2VHcm91cExvY2FsZXNXKExBTkdHUk9VUExPQ0FMRV9FTlVNUFJPQ1cgcExhbmdHcnBMY0VudW1Qcm9jLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExHUlBJRCBsZ3JwaWQsIERXT1JEIGR3RmxhZ3MsIExPTkdfUFRSIGxQYXJhbSkKewogICAgRU5VTUxBTkdVQUdFR1JPVVBMT0NBTEVfQ0FMTEJBQ0tTIGNhbGxiYWNrczsKCiAgICBUUkFDRSgiKCVwLDB4JTA4WCwweCUwOFgsMHglMDhsWClcbiIsIHBMYW5nR3JwTGNFbnVtUHJvYywgbGdycGlkLCBkd0ZsYWdzLCBsUGFyYW0pOwoKICAgIGNhbGxiYWNrcy5wcm9jQSAgID0gTlVMTDsKICAgIGNhbGxiYWNrcy5wcm9jVyAgID0gcExhbmdHcnBMY0VudW1Qcm9jOwogICAgY2FsbGJhY2tzLmR3RmxhZ3MgPSBkd0ZsYWdzOwogICAgY2FsbGJhY2tzLmxncnBpZCAgPSBsZ3JwaWQ7CiAgICBjYWxsYmFja3MubFBhcmFtICA9IGxQYXJhbTsKCiAgICByZXR1cm4gTkxTX0VudW1MYW5ndWFnZUdyb3VwTG9jYWxlcyggcExhbmdHcnBMY0VudW1Qcm9jID8gJmNhbGxiYWNrcyA6IE5VTEwgKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVN5c3RlbUdlb0lEICAgIChLRVJORUwzMi5AKQogKgogKiBDYWxsIGEgdXNlcnMgZnVuY3Rpb24gZm9yIGV2ZXJ5IGxvY2F0aW9uIGF2YWlsYWJsZSBvbiB0aGUgc3lzdGVtLgogKgogKiBQQVJBTVMKICogIGdlb2NsYXNzICAgICBbSV0gVHlwZSBvZiBpbmZvcm1hdGlvbiBkZXNpcmVkIChTWVNHRU9UWVBFIGVudW0gZnJvbSAid2lubmxzLmgiKQogKiAgcmVzZXJ2ZWQgICAgIFtJXSBSZXNlcnZlZCwgc2V0IHRvIDAKICogIHBHZW9FbnVtUHJvYyBbSV0gQ2FsbGJhY2sgZnVuY3Rpb24gdG8gY2FsbCBmb3IgZWFjaCBsb2NhdGlvbgogKgogKiBSRVRVUk5TCiAqICBTdWNjZXNzOiBUUlVFLgogKiAgRmFpbHVyZTogRkFMU0UuIFVzZSBHZXRMYXN0RXJyb3IoKSB0byBkZXRlcm1pbmUgdGhlIGNhdXNlLgogKi8KQk9PTCBXSU5BUEkgRW51bVN5c3RlbUdlb0lEKEdFT0NMQVNTIGdlb2NsYXNzLCBHRU9JRCByZXNlcnZlZCwgR0VPX0VOVU1QUk9DIHBHZW9FbnVtUHJvYykKewogICAgc3RhdGljIGNvbnN0IFdDSEFSIHN6Q291bnRyeUNvZGVWYWx1ZU5hbWVbXSA9IHsKICAgICAgJ0MnLCdvJywndScsJ24nLCd0JywncicsJ3knLCdDJywnbycsJ2QnLCdlJywnXDAnCiAgICB9OwogICAgV0NIQVIgc3pOdW1iZXJbMTBdOwogICAgSEFORExFIGhLZXk7CiAgICBVTE9ORyB1bEluZGV4ID0gMDsKCiAgICBUUkFDRSgiKDB4JTA4WCwweCUwOFgsJXApXG4iLCBnZW9jbGFzcywgcmVzZXJ2ZWQsIHBHZW9FbnVtUHJvYyk7CgogICAgaWYgKGdlb2NsYXNzICE9IEdFT0NMQVNTX05BVElPTiB8fCByZXNlcnZlZCB8fCAhcEdlb0VudW1Qcm9jKQogICAgewogICAgICAgIFNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGhLZXkgPSBOTFNfUmVnT3BlbktleSggMCwgc3pDb3VudHJ5TGlzdE5hbWUgKTsKCiAgICB3aGlsZSAoTkxTX1JlZ0VudW1TdWJLZXkoIGhLZXksIHVsSW5kZXgsIHN6TnVtYmVyLCBzaXplb2Yoc3pOdW1iZXIpICkpCiAgICB7CiAgICAgICAgQk9PTCBiQ29udGludWUgPSBUUlVFOwogICAgICAgIERXT1JEIGR3R2VvSWQ7CiAgICAgICAgSEFORExFIGhTdWJLZXkgPSBOTFNfUmVnT3BlbktleSggaEtleSwgc3pOdW1iZXIgKTsKCiAgICAgICAgaWYgKGhTdWJLZXkpCiAgICAgICAgewogICAgICAgICAgICBpZiAoTkxTX1JlZ0dldER3b3JkKCBoU3ViS2V5LCBzekNvdW50cnlDb2RlVmFsdWVOYW1lLCAmZHdHZW9JZCApKQogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICBUUkFDRSgiR290IGdlb2lkICVkXG4iLCBkd0dlb0lkKTsKCiAgICAgICAgICAgICAgICBpZiAoIXBHZW9FbnVtUHJvYyggZHdHZW9JZCApKQogICAgICAgICAgICAgICAgICAgIGJDb250aW51ZSA9IEZBTFNFOwogICAgICAgICAgICB9CgogICAgICAgICAgICBOdENsb3NlKCBoU3ViS2V5ICk7CiAgICAgICAgfQoKICAgICAgICBpZiAoIWJDb250aW51ZSkKICAgICAgICAgICAgYnJlYWs7CgogICAgICAgIHVsSW5kZXgrKzsKICAgIH0KCiAgICBpZiAoaEtleSkKICAgICAgICBOdENsb3NlKCBoS2V5ICk7CgogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEludmFsaWRhdGVOTFNDYWNoZSAgICAgICAgICAgKEtFUk5FTDMyLkApCiAqCiAqIEludmFsaWRhdGUgdGhlIGNhY2hlIG9mIE5MUyB2YWx1ZXMuCiAqCiAqIFBBUkFNUwogKiAgTm9uZS4KICoKICogUkVUVVJOUwogKiAgU3VjY2VzczogVFJVRS4KICogIEZhaWx1cmU6IEZBTFNFLgogKi8KQk9PTCBXSU5BUEkgSW52YWxpZGF0ZU5MU0NhY2hlKHZvaWQpCnsKICBGSVhNRSgiKCkgc3R1YlxuIik7CiAgcmV0dXJuIEZBTFNFOwp9CgovKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAqICAgICAgICAgICBHZXRVc2VyR2VvSUQgKEtFUk5FTDMyLkApCiAqLwpHRU9JRCBXSU5BUEkgR2V0VXNlckdlb0lEKCBHRU9DTEFTUyBHZW9DbGFzcyApCnsKICAgIEdFT0lEIHJldCA9IEdFT0lEX05PVF9BVkFJTEFCTEU7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgZ2VvV1tdID0geydHJywnZScsJ28nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIG5hdGlvbldbXSA9IHsnTicsJ2EnLCd0JywnaScsJ28nLCduJywwfTsKICAgIFdDSEFSIGJ1ZmZlcldbNDBdLCAqZW5kOwogICAgRFdPUkQgY291bnQ7CiAgICBIQU5ETEUgaGtleSwgaFN1YmtleSA9IDA7CiAgICBVTklDT0RFX1NUUklORyBrZXlXOwogICAgY29uc3QgS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKmluZm8gPSAoS0VZX1ZBTFVFX1BBUlRJQUxfSU5GT1JNQVRJT04gKilidWZmZXJXOwogICAgUnRsSW5pdFVuaWNvZGVTdHJpbmcoICZrZXlXLCBuYXRpb25XICk7CiAgICBjb3VudCA9IHNpemVvZihidWZmZXJXKTsKCiAgICBpZighKGhrZXkgPSBjcmVhdGVfcmVnaXN0cnlfa2V5KCkpKSByZXR1cm4gcmV0OwoKICAgIHN3aXRjaCggR2VvQ2xhc3MgKXsKICAgIGNhc2UgR0VPQ0xBU1NfTkFUSU9OOgogICAgICAgIGlmICgoaFN1YmtleSA9IE5MU19SZWdPcGVuS2V5KGhrZXksIGdlb1cpKSkKICAgICAgICB7CiAgICAgICAgICAgIGlmKChOdFF1ZXJ5VmFsdWVLZXkoaFN1YmtleSwgJmtleVcsIEtleVZhbHVlUGFydGlhbEluZm9ybWF0aW9uLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMUEJZVEUpYnVmZmVyVywgY291bnQsICZjb3VudCkgPT0gU1RBVFVTX1NVQ0NFU1MgKSAmJiBpbmZvLT5EYXRhTGVuZ3RoKQogICAgICAgICAgICAgICAgcmV0ID0gc3RydG9sVygoTFBDV1NUUilpbmZvLT5EYXRhLCAmZW5kLCAxMCk7CiAgICAgICAgfQogICAgICAgIGJyZWFrOwogICAgY2FzZSBHRU9DTEFTU19SRUdJT046CiAgICAgICAgRklYTUUoIkdFT0NMQVNTX1JFR0lPTiBub3QgaGFuZGxlZCB5ZXRcbiIpOwogICAgICAgIGJyZWFrOwogICAgfQoKICAgIE50Q2xvc2UoaGtleSk7CiAgICBpZiAoaFN1YmtleSkgTnRDbG9zZShoU3Via2V5KTsKICAgIHJldHVybiByZXQ7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIFNldFVzZXJHZW9JRCAoS0VSTkVMMzIuQCkKICovCkJPT0wgV0lOQVBJIFNldFVzZXJHZW9JRCggR0VPSUQgR2VvSUQgKQp7CiAgICBzdGF0aWMgY29uc3QgV0NIQVIgZ2VvV1tdID0geydHJywnZScsJ28nLDB9OwogICAgc3RhdGljIGNvbnN0IFdDSEFSIG5hdGlvbldbXSA9IHsnTicsJ2EnLCd0JywnaScsJ28nLCduJywwfTsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBmb3JtYXRXW10gPSB7JyUnLCdpJywwfTsKICAgIFVOSUNPREVfU1RSSU5HIG5hbWVXLGtleVc7CiAgICBXQ0hBUiBidWZmZXJXWzEwXTsKICAgIE9CSkVDVF9BVFRSSUJVVEVTIGF0dHI7CiAgICBIQU5ETEUgaGtleTsKCiAgICBpZighKGhrZXkgPSBjcmVhdGVfcmVnaXN0cnlfa2V5KCkpKSByZXR1cm4gRkFMU0U7CgogICAgYXR0ci5MZW5ndGggPSBzaXplb2YoYXR0cik7CiAgICBhdHRyLlJvb3REaXJlY3RvcnkgPSBoa2V5OwogICAgYXR0ci5PYmplY3ROYW1lID0gJm5hbWVXOwogICAgYXR0ci5BdHRyaWJ1dGVzID0gMDsKICAgIGF0dHIuU2VjdXJpdHlEZXNjcmlwdG9yID0gTlVMTDsKICAgIGF0dHIuU2VjdXJpdHlRdWFsaXR5T2ZTZXJ2aWNlID0gTlVMTDsKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAmbmFtZVcsIGdlb1cgKTsKICAgIFJ0bEluaXRVbmljb2RlU3RyaW5nKCAma2V5VywgbmF0aW9uVyApOwoKICAgIGlmIChOdENyZWF0ZUtleSggJmhrZXksIEtFWV9BTExfQUNDRVNTLCAmYXR0ciwgMCwgTlVMTCwgMCwgTlVMTCApICE9IFNUQVRVU19TVUNDRVNTKQoKICAgIHsKICAgICAgICBOdENsb3NlKGF0dHIuUm9vdERpcmVjdG9yeSk7CiAgICAgICAgcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIHNwcmludGZXKGJ1ZmZlclcsIGZvcm1hdFcsIEdlb0lEKTsKICAgIE50U2V0VmFsdWVLZXkoaGtleSwgJmtleVcsIDAsIFJFR19TWiwgYnVmZmVyVywgKHN0cmxlblcoYnVmZmVyVykgKyAxKSAqIHNpemVvZihXQ0hBUikpOwogICAgTnRDbG9zZShhdHRyLlJvb3REaXJlY3RvcnkpOwogICAgTnRDbG9zZShoa2V5KTsKICAgIHJldHVybiBUUlVFOwp9Cgp0eXBlZGVmIHN0cnVjdAp7CiAgICB1bmlvbgogICAgewogICAgICAgIFVJTEFOR1VBR0VfRU5VTVBST0NBIHByb2NBOwogICAgICAgIFVJTEFOR1VBR0VfRU5VTVBST0NXIHByb2NXOwogICAgfSB1OwogICAgRFdPUkQgZmxhZ3M7CiAgICBMT05HX1BUUiBwYXJhbTsKfSBFTlVNX1VJTEFOR19DQUxMQkFDSzsKCnN0YXRpYyBCT09MIENBTExCQUNLIGVudW1fdWlsYW5nX3Byb2NfYSggSE1PRFVMRSBoTW9kdWxlLCBMUENTVFIgdHlwZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBMUENTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExPTkdfUFRSIGxQYXJhbSApCnsKICAgIEVOVU1fVUlMQU5HX0NBTExCQUNLICplbnVtX3VpbGFuZyA9IChFTlVNX1VJTEFOR19DQUxMQkFDSyAqKWxQYXJhbTsKICAgIGNoYXIgYnVmWzIwXTsKCiAgICBzcHJpbnRmKGJ1ZiwgIiUwOHgiLCAoVUlOVClMYW5nSUQpOwogICAgcmV0dXJuIGVudW1fdWlsYW5nLT51LnByb2NBKCBidWYsIGVudW1fdWlsYW5nLT5wYXJhbSApOwp9CgpzdGF0aWMgQk9PTCBDQUxMQkFDSyBlbnVtX3VpbGFuZ19wcm9jX3coIEhNT0RVTEUgaE1vZHVsZSwgTFBDV1NUUiB0eXBlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExQQ1dTVFIgbmFtZSwgV09SRCBMYW5nSUQsIExPTkdfUFRSIGxQYXJhbSApCnsKICAgIHN0YXRpYyBjb25zdCBXQ0hBUiBmb3JtYXRXW10gPSB7JyUnLCcwJywnOCcsJ3gnLDB9OwogICAgRU5VTV9VSUxBTkdfQ0FMTEJBQ0sgKmVudW1fdWlsYW5nID0gKEVOVU1fVUlMQU5HX0NBTExCQUNLICopbFBhcmFtOwogICAgV0NIQVIgYnVmWzIwXTsKCiAgICBzcHJpbnRmVyggYnVmLCBmb3JtYXRXLCAoVUlOVClMYW5nSUQgKTsKICAgIHJldHVybiBlbnVtX3VpbGFuZy0+dS5wcm9jVyggYnVmLCBlbnVtX3VpbGFuZy0+cGFyYW0gKTsKfQoKLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgogKiAgICAgICAgICAgRW51bVVJTGFuZ3VhZ2VzQSAoS0VSTkVMMzIuQCkKICovCkJPT0wgV0lOQVBJIEVudW1VSUxhbmd1YWdlc0EoVUlMQU5HVUFHRV9FTlVNUFJPQ0EgcFVJTGFuZ0VudW1Qcm9jLCBEV09SRCBkd0ZsYWdzLCBMT05HX1BUUiBsUGFyYW0pCnsKICAgIEVOVU1fVUlMQU5HX0NBTExCQUNLIGVudW1fdWlsYW5nOwoKICAgIFRSQUNFKCIlcCwgJXgsICVseFxuIiwgcFVJTGFuZ0VudW1Qcm9jLCBkd0ZsYWdzLCBsUGFyYW0pOwoKICAgIGlmKCFwVUlMYW5nRW51bVByb2MpIHsKCVNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX1BBUkFNRVRFUik7CglyZXR1cm4gRkFMU0U7CiAgICB9CiAgICBpZihkd0ZsYWdzKSB7CglTZXRMYXN0RXJyb3IoRVJST1JfSU5WQUxJRF9GTEFHUyk7CglyZXR1cm4gRkFMU0U7CiAgICB9CgogICAgZW51bV91aWxhbmcudS5wcm9jQSA9IHBVSUxhbmdFbnVtUHJvYzsKICAgIGVudW1fdWlsYW5nLmZsYWdzID0gZHdGbGFnczsKICAgIGVudW1fdWlsYW5nLnBhcmFtID0gbFBhcmFtOwoKICAgIEVudW1SZXNvdXJjZUxhbmd1YWdlc0EoIGtlcm5lbDMyX2hhbmRsZSwgKExQQ1NUUilSVF9TVFJJTkcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBDU1RSKUxPQ0FMRV9JTEFOR1VBR0UsIGVudW1fdWlsYW5nX3Byb2NfYSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIChMT05HX1BUUikmZW51bV91aWxhbmcpOwogICAgcmV0dXJuIFRSVUU7Cn0KCi8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKICogICAgICAgICAgIEVudW1VSUxhbmd1YWdlc1cgKEtFUk5FTDMyLkApCiAqLwpCT09MIFdJTkFQSSBFbnVtVUlMYW5ndWFnZXNXKFVJTEFOR1VBR0VfRU5VTVBST0NXIHBVSUxhbmdFbnVtUHJvYywgRFdPUkQgZHdGbGFncywgTE9OR19QVFIgbFBhcmFtKQp7CiAgICBFTlVNX1VJTEFOR19DQUxMQkFDSyBlbnVtX3VpbGFuZzsKCiAgICBUUkFDRSgiJXAsICV4LCAlbHhcbiIsIHBVSUxhbmdFbnVtUHJvYywgZHdGbGFncywgbFBhcmFtKTsKCgogICAgaWYoIXBVSUxhbmdFbnVtUHJvYykgewoJU2V0TGFzdEVycm9yKEVSUk9SX0lOVkFMSURfUEFSQU1FVEVSKTsKCXJldHVybiBGQUxTRTsKICAgIH0KICAgIGlmKGR3RmxhZ3MpIHsKCVNldExhc3RFcnJvcihFUlJPUl9JTlZBTElEX0ZMQUdTKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBlbnVtX3VpbGFuZy51LnByb2NXID0gcFVJTGFuZ0VudW1Qcm9jOwogICAgZW51bV91aWxhbmcuZmxhZ3MgPSBkd0ZsYWdzOwogICAgZW51bV91aWxhbmcucGFyYW0gPSBsUGFyYW07CgogICAgRW51bVJlc291cmNlTGFuZ3VhZ2VzVygga2VybmVsMzJfaGFuZGxlLCAoTFBDV1NUUilSVF9TVFJJTkcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTFBDV1NUUilMT0NBTEVfSUxBTkdVQUdFLCBlbnVtX3VpbGFuZ19wcm9jX3csCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoTE9OR19QVFIpJmVudW1fdWlsYW5nKTsKICAgIHJldHVybiBUUlVFOwp9CgpJTlQgV0lOQVBJIEdldEdlb0luZm9XKEdFT0lEIEdlb0lkLCBHRU9UWVBFIEdlb1R5cGUsIExQV1NUUiBscEdlb0RhdGEsIAogICAgICAgICAgICAgICAgaW50IGNjaERhdGEsIExBTkdJRCBsYW5ndWFnZSkKewogICAgRklYTUUoIiVkICVkICVwICVkICVkXG4iLCBHZW9JZCwgR2VvVHlwZSwgbHBHZW9EYXRhLCBjY2hEYXRhLCBsYW5ndWFnZSk7CiAgICByZXR1cm4gMDsKfQoKSU5UIFdJTkFQSSBHZXRHZW9JbmZvQShHRU9JRCBHZW9JZCwgR0VPVFlQRSBHZW9UeXBlLCBMUFNUUiBscEdlb0RhdGEsIAogICAgICAgICAgICAgICAgaW50IGNjaERhdGEsIExBTkdJRCBsYW5ndWFnZSkKewogICAgRklYTUUoIiVkICVkICVwICVkICVkXG4iLCBHZW9JZCwgR2VvVHlwZSwgbHBHZW9EYXRhLCBjY2hEYXRhLCBsYW5ndWFnZSk7CiAgICByZXR1cm4gMDsKfQo=