LyogbW9kdWxlLmMgLSB0aGUgbW9kdWxlIGl0c2VsZgogKgogKiBDb3B5cmlnaHQgKEMpIDIwMDQtMjAwNyBHZXJoYXJkIEjkcmluZyA8Z2hAZ2hhZXJpbmcuZGU+CiAqCiAqIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHB5c3FsaXRlLgogKgogKiBUaGlzIHNvZnR3YXJlIGlzIHByb3ZpZGVkICdhcy1pcycsIHdpdGhvdXQgYW55IGV4cHJlc3Mgb3IgaW1wbGllZAogKiB3YXJyYW50eS4gIEluIG5vIGV2ZW50IHdpbGwgdGhlIGF1dGhvcnMgYmUgaGVsZCBsaWFibGUgZm9yIGFueSBkYW1hZ2VzCiAqIGFyaXNpbmcgZnJvbSB0aGUgdXNlIG9mIHRoaXMgc29mdHdhcmUuCiAqCiAqIFBlcm1pc3Npb24gaXMgZ3JhbnRlZCB0byBhbnlvbmUgdG8gdXNlIHRoaXMgc29mdHdhcmUgZm9yIGFueSBwdXJwb3NlLAogKiBpbmNsdWRpbmcgY29tbWVyY2lhbCBhcHBsaWNhdGlvbnMsIGFuZCB0byBhbHRlciBpdCBhbmQgcmVkaXN0cmlidXRlIGl0CiAqIGZyZWVseSwgc3ViamVjdCB0byB0aGUgZm9sbG93aW5nIHJlc3RyaWN0aW9uczoKICoKICogMS4gVGhlIG9yaWdpbiBvZiB0aGlzIHNvZnR3YXJlIG11c3Qgbm90IGJlIG1pc3JlcHJlc2VudGVkOyB5b3UgbXVzdCBub3QKICogICAgY2xhaW0gdGhhdCB5b3Ugd3JvdGUgdGhlIG9yaWdpbmFsIHNvZnR3YXJlLiBJZiB5b3UgdXNlIHRoaXMgc29mdHdhcmUKICogICAgaW4gYSBwcm9kdWN0LCBhbiBhY2tub3dsZWRnbWVudCBpbiB0aGUgcHJvZHVjdCBkb2N1bWVudGF0aW9uIHdvdWxkIGJlCiAqICAgIGFwcHJlY2lhdGVkIGJ1dCBpcyBub3QgcmVxdWlyZWQuCiAqIDIuIEFsdGVyZWQgc291cmNlIHZlcnNpb25zIG11c3QgYmUgcGxhaW5seSBtYXJrZWQgYXMgc3VjaCwgYW5kIG11c3Qgbm90IGJlCiAqICAgIG1pc3JlcHJlc2VudGVkIGFzIGJlaW5nIHRoZSBvcmlnaW5hbCBzb2Z0d2FyZS4KICogMy4gVGhpcyBub3RpY2UgbWF5IG5vdCBiZSByZW1vdmVkIG9yIGFsdGVyZWQgZnJvbSBhbnkgc291cmNlIGRpc3RyaWJ1dGlvbi4KICovCgojaW5jbHVkZSAiY29ubmVjdGlvbi5oIgojaW5jbHVkZSAic3RhdGVtZW50LmgiCiNpbmNsdWRlICJjdXJzb3IuaCIKI2luY2x1ZGUgImNhY2hlLmgiCiNpbmNsdWRlICJwcmVwYXJlX3Byb3RvY29sLmgiCiNpbmNsdWRlICJtaWNyb3Byb3RvY29scy5oIgojaW5jbHVkZSAicm93LmgiCgojaWYgU1FMSVRFX1ZFUlNJT05fTlVNQkVSID49IDMwMDMwMDMKI2RlZmluZSBIQVZFX1NIQVJFRF9DQUNIRQojZW5kaWYKCi8qIHN0YXRpYyBvYmplY3RzIGF0IG1vZHVsZS1sZXZlbCAqLwoKUHlPYmplY3QqIHB5c3FsaXRlX0Vycm9yLCAqcHlzcWxpdGVfV2FybmluZywgKnB5c3FsaXRlX0ludGVyZmFjZUVycm9yLCAqcHlzcWxpdGVfRGF0YWJhc2VFcnJvciwKICAgICpweXNxbGl0ZV9JbnRlcm5hbEVycm9yLCAqcHlzcWxpdGVfT3BlcmF0aW9uYWxFcnJvciwgKnB5c3FsaXRlX1Byb2dyYW1taW5nRXJyb3IsCiAgICAqcHlzcWxpdGVfSW50ZWdyaXR5RXJyb3IsICpweXNxbGl0ZV9EYXRhRXJyb3IsICpweXNxbGl0ZV9Ob3RTdXBwb3J0ZWRFcnJvciwgKnB5c3FsaXRlX09wdGltaXplZFVuaWNvZGU7CgpQeU9iamVjdCogY29udmVydGVyczsKaW50IF9lbmFibGVfY2FsbGJhY2tfdHJhY2ViYWNrczsKaW50IHB5c3FsaXRlX0Jhc2VUeXBlQWRhcHRlZDsKCnN0YXRpYyBQeU9iamVjdCogbW9kdWxlX2Nvbm5lY3QoUHlPYmplY3QqIHNlbGYsIFB5T2JqZWN0KiBhcmdzLCBQeU9iamVjdCoKICAgICAgICBrd2FyZ3MpCnsKICAgIC8qIFB5dGhvbiBzZWVtcyB0byBoYXZlIG5vIHdheSBvZiBleHRyYWN0aW5nIGEgc2luZ2xlIGtleXdvcmQtYXJnIGF0CiAgICAgKiBDLWxldmVsLCBzbyB0aGlzIGNvZGUgaXMgcmVkdW5kYW50IHdpdGggdGhlIG9uZSBpbiBjb25uZWN0aW9uX2luaXQgaW4KICAgICAqIGNvbm5lY3Rpb24uYyBhbmQgbXVzdCBhbHdheXMgYmUgY29waWVkIGZyb20gdGhlcmUgLi4uICovCgogICAgc3RhdGljIGNoYXIgKmt3bGlzdFtdID0geyJkYXRhYmFzZSIsICJ0aW1lb3V0IiwgImRldGVjdF90eXBlcyIsICJpc29sYXRpb25fbGV2ZWwiLCAiY2hlY2tfc2FtZV90aHJlYWQiLCAiZmFjdG9yeSIsICJjYWNoZWRfc3RhdGVtZW50cyIsIE5VTEwsIE5VTEx9OwogICAgUHlPYmplY3QqIGRhdGFiYXNlOwogICAgaW50IGRldGVjdF90eXBlcyA9IDA7CiAgICBQeU9iamVjdCogaXNvbGF0aW9uX2xldmVsOwogICAgUHlPYmplY3QqIGZhY3RvcnkgPSBOVUxMOwogICAgaW50IGNoZWNrX3NhbWVfdGhyZWFkID0gMTsKICAgIGludCBjYWNoZWRfc3RhdGVtZW50czsKICAgIGRvdWJsZSB0aW1lb3V0ID0gNS4wOwoKICAgIFB5T2JqZWN0KiByZXN1bHQ7CgogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlQW5kS2V5d29yZHMoYXJncywga3dhcmdzLCAiT3xkaU9pT2kiLCBrd2xpc3QsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmZGF0YWJhc2UsICZ0aW1lb3V0LCAmZGV0ZWN0X3R5cGVzLCAmaXNvbGF0aW9uX2xldmVsLCAmY2hlY2tfc2FtZV90aHJlYWQsICZmYWN0b3J5LCAmY2FjaGVkX3N0YXRlbWVudHMpKQogICAgewogICAgICAgIHJldHVybiBOVUxMOyAKICAgIH0KCiAgICBpZiAoZmFjdG9yeSA9PSBOVUxMKSB7CiAgICAgICAgZmFjdG9yeSA9IChQeU9iamVjdCopJnB5c3FsaXRlX0Nvbm5lY3Rpb25UeXBlOwogICAgfQoKICAgIHJlc3VsdCA9IFB5T2JqZWN0X0NhbGwoZmFjdG9yeSwgYXJncywga3dhcmdzKTsKCiAgICByZXR1cm4gcmVzdWx0Owp9CgpzdGF0aWMgUHlPYmplY3QqIG1vZHVsZV9jb21wbGV0ZShQeU9iamVjdCogc2VsZiwgUHlPYmplY3QqIGFyZ3MsIFB5T2JqZWN0KgogICAgICAgIGt3YXJncykKewogICAgc3RhdGljIGNoYXIgKmt3bGlzdFtdID0geyJzdGF0ZW1lbnQiLCBOVUxMLCBOVUxMfTsKICAgIGNoYXIqIHN0YXRlbWVudDsKCiAgICBQeU9iamVjdCogcmVzdWx0OwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZUFuZEtleXdvcmRzKGFyZ3MsIGt3YXJncywgInMiLCBrd2xpc3QsICZzdGF0ZW1lbnQpKQogICAgewogICAgICAgIHJldHVybiBOVUxMOyAKICAgIH0KCiAgICBpZiAoc3FsaXRlM19jb21wbGV0ZShzdGF0ZW1lbnQpKSB7CiAgICAgICAgcmVzdWx0ID0gUHlfVHJ1ZTsKICAgIH0gZWxzZSB7CiAgICAgICAgcmVzdWx0ID0gUHlfRmFsc2U7CiAgICB9CgogICAgUHlfSU5DUkVGKHJlc3VsdCk7CgogICAgcmV0dXJuIHJlc3VsdDsKfQoKI2lmZGVmIEhBVkVfU0hBUkVEX0NBQ0hFCnN0YXRpYyBQeU9iamVjdCogbW9kdWxlX2VuYWJsZV9zaGFyZWRfY2FjaGUoUHlPYmplY3QqIHNlbGYsIFB5T2JqZWN0KiBhcmdzLCBQeU9iamVjdCoKICAgICAgICBrd2FyZ3MpCnsKICAgIHN0YXRpYyBjaGFyICprd2xpc3RbXSA9IHsiZG9fZW5hYmxlIiwgTlVMTCwgTlVMTH07CiAgICBpbnQgZG9fZW5hYmxlOwogICAgaW50IHJjOwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZUFuZEtleXdvcmRzKGFyZ3MsIGt3YXJncywgImkiLCBrd2xpc3QsICZkb19lbmFibGUpKQogICAgewogICAgICAgIHJldHVybiBOVUxMOyAKICAgIH0KCiAgICByYyA9IHNxbGl0ZTNfZW5hYmxlX3NoYXJlZF9jYWNoZShkb19lbmFibGUpOwoKICAgIGlmIChyYyAhPSBTUUxJVEVfT0spIHsKICAgICAgICBQeUVycl9TZXRTdHJpbmcocHlzcWxpdGVfT3BlcmF0aW9uYWxFcnJvciwgIkNoYW5naW5nIHRoZSBzaGFyZWRfY2FjaGUgZmxhZyBmYWlsZWQiKTsKICAgICAgICByZXR1cm4gTlVMTDsKICAgIH0gZWxzZSB7CiAgICAgICAgUHlfSU5DUkVGKFB5X05vbmUpOwogICAgICAgIHJldHVybiBQeV9Ob25lOwogICAgfQp9CiNlbmRpZiAvKiBIQVZFX1NIQVJFRF9DQUNIRSAqLwoKc3RhdGljIFB5T2JqZWN0KiBtb2R1bGVfcmVnaXN0ZXJfYWRhcHRlcihQeU9iamVjdCogc2VsZiwgUHlPYmplY3QqIGFyZ3MsIFB5T2JqZWN0KiBrd2FyZ3MpCnsKICAgIFB5VHlwZU9iamVjdCogdHlwZTsKICAgIFB5T2JqZWN0KiBjYXN0ZXI7CgogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJPTyIsICZ0eXBlLCAmY2FzdGVyKSkgewogICAgICAgIHJldHVybiBOVUxMOwogICAgfQoKICAgIC8qIGEgYmFzaWMgdHlwZSBpcyBhZGFwdGVkOyB0aGVyZSdzIGEgcGVyZm9ybWFuY2Ugb3B0aW1pemF0aW9uIGlmIHRoYXQncyBub3QgdGhlIGNhc2UKICAgICAqICg5OSAlIG9mIGFsbCB1c2FnZXMpICovCiAgICBpZiAodHlwZSA9PSAmUHlJbnRfVHlwZSB8fCB0eXBlID09ICZQeUxvbmdfVHlwZSB8fCB0eXBlID09ICZQeUZsb2F0X1R5cGUKICAgICAgICAgICAgfHwgdHlwZSA9PSAmUHlTdHJpbmdfVHlwZSB8fCB0eXBlID09ICZQeVVuaWNvZGVfVHlwZSB8fCB0eXBlID09ICZQeUJ1ZmZlcl9UeXBlKSB7CiAgICAgICAgcHlzcWxpdGVfQmFzZVR5cGVBZGFwdGVkID0gMTsKICAgIH0KCiAgICBtaWNyb3Byb3RvY29sc19hZGQodHlwZSwgKFB5T2JqZWN0KikmcHlzcWxpdGVfUHJlcGFyZVByb3RvY29sVHlwZSwgY2FzdGVyKTsKCiAgICBQeV9JTkNSRUYoUHlfTm9uZSk7CiAgICByZXR1cm4gUHlfTm9uZTsKfQoKc3RhdGljIFB5T2JqZWN0KiBtb2R1bGVfcmVnaXN0ZXJfY29udmVydGVyKFB5T2JqZWN0KiBzZWxmLCBQeU9iamVjdCogYXJncywgUHlPYmplY3QqIGt3YXJncykKewogICAgUHlPYmplY3QqIG9yaWdfbmFtZTsKICAgIFB5T2JqZWN0KiBuYW1lID0gTlVMTDsKICAgIFB5T2JqZWN0KiBjYWxsYWJsZTsKICAgIFB5T2JqZWN0KiByZXR2YWwgPSBOVUxMOwoKICAgIGlmICghUHlBcmdfUGFyc2VUdXBsZShhcmdzLCAiU08iLCAmb3JpZ19uYW1lLCAmY2FsbGFibGUpKSB7CiAgICAgICAgcmV0dXJuIE5VTEw7CiAgICB9CgogICAgLyogY29udmVydCB0aGUgbmFtZSB0byB1cHBlciBjYXNlICovCiAgICBuYW1lID0gUHlPYmplY3RfQ2FsbE1ldGhvZChvcmlnX25hbWUsICJ1cHBlciIsICIiKTsKICAgIGlmICghbmFtZSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CgogICAgaWYgKFB5RGljdF9TZXRJdGVtKGNvbnZlcnRlcnMsIG5hbWUsIGNhbGxhYmxlKSAhPSAwKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KCiAgICBQeV9JTkNSRUYoUHlfTm9uZSk7CiAgICByZXR2YWwgPSBQeV9Ob25lOwplcnJvcjoKICAgIFB5X1hERUNSRUYobmFtZSk7CiAgICByZXR1cm4gcmV0dmFsOwp9CgpzdGF0aWMgUHlPYmplY3QqIGVuYWJsZV9jYWxsYmFja190cmFjZWJhY2tzKFB5T2JqZWN0KiBzZWxmLCBQeU9iamVjdCogYXJncywgUHlPYmplY3QqIGt3YXJncykKewogICAgaWYgKCFQeUFyZ19QYXJzZVR1cGxlKGFyZ3MsICJpIiwgJl9lbmFibGVfY2FsbGJhY2tfdHJhY2ViYWNrcykpIHsKICAgICAgICByZXR1cm4gTlVMTDsKICAgIH0KCiAgICBQeV9JTkNSRUYoUHlfTm9uZSk7CiAgICByZXR1cm4gUHlfTm9uZTsKfQoKc3RhdGljIHZvaWQgY29udmVydGVyc19pbml0KFB5T2JqZWN0KiBkaWN0KQp7CiAgICBjb252ZXJ0ZXJzID0gUHlEaWN0X05ldygpOwogICAgaWYgKCFjb252ZXJ0ZXJzKSB7CiAgICAgICAgcmV0dXJuOwogICAgfQoKICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJjb252ZXJ0ZXJzIiwgY29udmVydGVycyk7Cn0KCnN0YXRpYyBQeU1ldGhvZERlZiBtb2R1bGVfbWV0aG9kc1tdID0gewogICAgeyJjb25uZWN0IiwgIChQeUNGdW5jdGlvbiltb2R1bGVfY29ubmVjdCwgIE1FVEhfVkFSQVJHU3xNRVRIX0tFWVdPUkRTLCBQeURvY19TVFIoIkNyZWF0ZXMgYSBjb25uZWN0aW9uLiIpfSwKICAgIHsiY29tcGxldGVfc3RhdGVtZW50IiwgIChQeUNGdW5jdGlvbiltb2R1bGVfY29tcGxldGUsICBNRVRIX1ZBUkFSR1N8TUVUSF9LRVlXT1JEUywgUHlEb2NfU1RSKCJDaGVja3MgaWYgYSBzdHJpbmcgY29udGFpbnMgYSBjb21wbGV0ZSBTUUwgc3RhdGVtZW50LiBOb24tc3RhbmRhcmQuIil9LAojaWZkZWYgSEFWRV9TSEFSRURfQ0FDSEUKICAgIHsiZW5hYmxlX3NoYXJlZF9jYWNoZSIsICAoUHlDRnVuY3Rpb24pbW9kdWxlX2VuYWJsZV9zaGFyZWRfY2FjaGUsICBNRVRIX1ZBUkFSR1N8TUVUSF9LRVlXT1JEUywgUHlEb2NfU1RSKCJFbmFibGUgb3IgZGlzYWJsZSBzaGFyZWQgY2FjaGUgbW9kZSBmb3IgdGhlIGNhbGxpbmcgdGhyZWFkLiBFeHBlcmltZW50YWwvTm9uLXN0YW5kYXJkLiIpfSwKI2VuZGlmCiAgICB7InJlZ2lzdGVyX2FkYXB0ZXIiLCAoUHlDRnVuY3Rpb24pbW9kdWxlX3JlZ2lzdGVyX2FkYXB0ZXIsIE1FVEhfVkFSQVJHUywgUHlEb2NfU1RSKCJSZWdpc3RlcnMgYW4gYWRhcHRlciB3aXRoIHB5c3FsaXRlJ3MgYWRhcHRlciByZWdpc3RyeS4gTm9uLXN0YW5kYXJkLiIpfSwKICAgIHsicmVnaXN0ZXJfY29udmVydGVyIiwgKFB5Q0Z1bmN0aW9uKW1vZHVsZV9yZWdpc3Rlcl9jb252ZXJ0ZXIsIE1FVEhfVkFSQVJHUywgUHlEb2NfU1RSKCJSZWdpc3RlcnMgYSBjb252ZXJ0ZXIgd2l0aCBweXNxbGl0ZS4gTm9uLXN0YW5kYXJkLiIpfSwKICAgIHsiYWRhcHQiLCAgKFB5Q0Z1bmN0aW9uKXBzeWNvX21pY3JvcHJvdG9jb2xzX2FkYXB0LCBNRVRIX1ZBUkFSR1MsIHBzeWNvX21pY3JvcHJvdG9jb2xzX2FkYXB0X2RvY30sCiAgICB7ImVuYWJsZV9jYWxsYmFja190cmFjZWJhY2tzIiwgIChQeUNGdW5jdGlvbillbmFibGVfY2FsbGJhY2tfdHJhY2ViYWNrcywgTUVUSF9WQVJBUkdTLCBQeURvY19TVFIoIkVuYWJsZSBvciBkaXNhYmxlIGNhbGxiYWNrIGZ1bmN0aW9ucyB0aHJvd2luZyBlcnJvcnMgdG8gc3RkZXJyLiIpfSwKICAgIHtOVUxMLCBOVUxMfQp9OwoKc3RydWN0IF9JbnRDb25zdGFudFBhaXIgewogICAgY2hhciogY29uc3RhbnRfbmFtZTsKICAgIGludCBjb25zdGFudF92YWx1ZTsKfTsKCnR5cGVkZWYgc3RydWN0IF9JbnRDb25zdGFudFBhaXIgSW50Q29uc3RhbnRQYWlyOwoKc3RhdGljIEludENvbnN0YW50UGFpciBfaW50X2NvbnN0YW50c1tdID0gewogICAgeyJQQVJTRV9ERUNMVFlQRVMiLCBQQVJTRV9ERUNMVFlQRVN9LAogICAgeyJQQVJTRV9DT0xOQU1FUyIsIFBBUlNFX0NPTE5BTUVTfSwKCiAgICB7IlNRTElURV9PSyIsIFNRTElURV9PS30sCiAgICB7IlNRTElURV9ERU5ZIiwgU1FMSVRFX0RFTll9LAogICAgeyJTUUxJVEVfSUdOT1JFIiwgU1FMSVRFX0lHTk9SRX0sCiAgICB7IlNRTElURV9DUkVBVEVfSU5ERVgiLCBTUUxJVEVfQ1JFQVRFX0lOREVYfSwKICAgIHsiU1FMSVRFX0NSRUFURV9UQUJMRSIsIFNRTElURV9DUkVBVEVfVEFCTEV9LAogICAgeyJTUUxJVEVfQ1JFQVRFX1RFTVBfSU5ERVgiLCBTUUxJVEVfQ1JFQVRFX1RFTVBfSU5ERVh9LAogICAgeyJTUUxJVEVfQ1JFQVRFX1RFTVBfVEFCTEUiLCBTUUxJVEVfQ1JFQVRFX1RFTVBfVEFCTEV9LAogICAgeyJTUUxJVEVfQ1JFQVRFX1RFTVBfVFJJR0dFUiIsIFNRTElURV9DUkVBVEVfVEVNUF9UUklHR0VSfSwKICAgIHsiU1FMSVRFX0NSRUFURV9URU1QX1ZJRVciLCBTUUxJVEVfQ1JFQVRFX1RFTVBfVklFV30sCiAgICB7IlNRTElURV9DUkVBVEVfVFJJR0dFUiIsIFNRTElURV9DUkVBVEVfVFJJR0dFUn0sCiAgICB7IlNRTElURV9DUkVBVEVfVklFVyIsIFNRTElURV9DUkVBVEVfVklFV30sCiAgICB7IlNRTElURV9ERUxFVEUiLCBTUUxJVEVfREVMRVRFfSwKICAgIHsiU1FMSVRFX0RST1BfSU5ERVgiLCBTUUxJVEVfRFJPUF9JTkRFWH0sCiAgICB7IlNRTElURV9EUk9QX1RBQkxFIiwgU1FMSVRFX0RST1BfVEFCTEV9LAogICAgeyJTUUxJVEVfRFJPUF9URU1QX0lOREVYIiwgU1FMSVRFX0RST1BfVEVNUF9JTkRFWH0sCiAgICB7IlNRTElURV9EUk9QX1RFTVBfVEFCTEUiLCBTUUxJVEVfRFJPUF9URU1QX1RBQkxFfSwKICAgIHsiU1FMSVRFX0RST1BfVEVNUF9UUklHR0VSIiwgU1FMSVRFX0RST1BfVEVNUF9UUklHR0VSfSwKICAgIHsiU1FMSVRFX0RST1BfVEVNUF9WSUVXIiwgU1FMSVRFX0RST1BfVEVNUF9WSUVXfSwKICAgIHsiU1FMSVRFX0RST1BfVFJJR0dFUiIsIFNRTElURV9EUk9QX1RSSUdHRVJ9LAogICAgeyJTUUxJVEVfRFJPUF9WSUVXIiwgU1FMSVRFX0RST1BfVklFV30sCiAgICB7IlNRTElURV9JTlNFUlQiLCBTUUxJVEVfSU5TRVJUfSwKICAgIHsiU1FMSVRFX1BSQUdNQSIsIFNRTElURV9QUkFHTUF9LAogICAgeyJTUUxJVEVfUkVBRCIsIFNRTElURV9SRUFEfSwKICAgIHsiU1FMSVRFX1NFTEVDVCIsIFNRTElURV9TRUxFQ1R9LAogICAgeyJTUUxJVEVfVFJBTlNBQ1RJT04iLCBTUUxJVEVfVFJBTlNBQ1RJT059LAogICAgeyJTUUxJVEVfVVBEQVRFIiwgU1FMSVRFX1VQREFURX0sCiAgICB7IlNRTElURV9BVFRBQ0giLCBTUUxJVEVfQVRUQUNIfSwKICAgIHsiU1FMSVRFX0RFVEFDSCIsIFNRTElURV9ERVRBQ0h9LAojaWYgU1FMSVRFX1ZFUlNJT05fTlVNQkVSID49IDMwMDIwMDEKICAgIHsiU1FMSVRFX0FMVEVSX1RBQkxFIiwgU1FMSVRFX0FMVEVSX1RBQkxFfSwKICAgIHsiU1FMSVRFX1JFSU5ERVgiLCBTUUxJVEVfUkVJTkRFWH0sCiNlbmRpZgojaWYgU1FMSVRFX1ZFUlNJT05fTlVNQkVSID49IDMwMDMwMDAKICAgIHsiU1FMSVRFX0FOQUxZWkUiLCBTUUxJVEVfQU5BTFlaRX0sCiNlbmRpZgogICAgeyhjaGFyKilOVUxMLCAwfQp9OwoKUHlNT0RJTklUX0ZVTkMgaW5pdF9zcWxpdGUzKHZvaWQpCnsKICAgIFB5T2JqZWN0ICptb2R1bGUsICpkaWN0OwogICAgUHlPYmplY3QgKnRtcF9vYmo7CiAgICBpbnQgaTsKCiAgICBtb2R1bGUgPSBQeV9Jbml0TW9kdWxlKCJfc3FsaXRlMyIsIG1vZHVsZV9tZXRob2RzKTsKCiAgICBpZiAoIW1vZHVsZSB8fAogICAgICAgIChweXNxbGl0ZV9yb3dfc2V0dXBfdHlwZXMoKSA8IDApIHx8CiAgICAgICAgKHB5c3FsaXRlX2N1cnNvcl9zZXR1cF90eXBlcygpIDwgMCkgfHwKICAgICAgICAocHlzcWxpdGVfY29ubmVjdGlvbl9zZXR1cF90eXBlcygpIDwgMCkgfHwKICAgICAgICAocHlzcWxpdGVfY2FjaGVfc2V0dXBfdHlwZXMoKSA8IDApIHx8CiAgICAgICAgKHB5c3FsaXRlX3N0YXRlbWVudF9zZXR1cF90eXBlcygpIDwgMCkgfHwKICAgICAgICAocHlzcWxpdGVfcHJlcGFyZV9wcm90b2NvbF9zZXR1cF90eXBlcygpIDwgMCkKICAgICAgICkgewogICAgICAgIHJldHVybjsKICAgIH0KCiAgICBQeV9JTkNSRUYoJnB5c3FsaXRlX0Nvbm5lY3Rpb25UeXBlKTsKICAgIFB5TW9kdWxlX0FkZE9iamVjdChtb2R1bGUsICJDb25uZWN0aW9uIiwgKFB5T2JqZWN0KikgJnB5c3FsaXRlX0Nvbm5lY3Rpb25UeXBlKTsKICAgIFB5X0lOQ1JFRigmcHlzcWxpdGVfQ3Vyc29yVHlwZSk7CiAgICBQeU1vZHVsZV9BZGRPYmplY3QobW9kdWxlLCAiQ3Vyc29yIiwgKFB5T2JqZWN0KikgJnB5c3FsaXRlX0N1cnNvclR5cGUpOwogICAgUHlfSU5DUkVGKCZweXNxbGl0ZV9DYWNoZVR5cGUpOwogICAgUHlNb2R1bGVfQWRkT2JqZWN0KG1vZHVsZSwgIlN0YXRlbWVudCIsIChQeU9iamVjdCopJnB5c3FsaXRlX1N0YXRlbWVudFR5cGUpOwogICAgUHlfSU5DUkVGKCZweXNxbGl0ZV9TdGF0ZW1lbnRUeXBlKTsKICAgIFB5TW9kdWxlX0FkZE9iamVjdChtb2R1bGUsICJDYWNoZSIsIChQeU9iamVjdCopICZweXNxbGl0ZV9DYWNoZVR5cGUpOwogICAgUHlfSU5DUkVGKCZweXNxbGl0ZV9QcmVwYXJlUHJvdG9jb2xUeXBlKTsKICAgIFB5TW9kdWxlX0FkZE9iamVjdChtb2R1bGUsICJQcmVwYXJlUHJvdG9jb2wiLCAoUHlPYmplY3QqKSAmcHlzcWxpdGVfUHJlcGFyZVByb3RvY29sVHlwZSk7CiAgICBQeV9JTkNSRUYoJnB5c3FsaXRlX1Jvd1R5cGUpOwogICAgUHlNb2R1bGVfQWRkT2JqZWN0KG1vZHVsZSwgIlJvdyIsIChQeU9iamVjdCopICZweXNxbGl0ZV9Sb3dUeXBlKTsKCiAgICBpZiAoIShkaWN0ID0gUHlNb2R1bGVfR2V0RGljdChtb2R1bGUpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CgogICAgLyoqKiBDcmVhdGUgREItQVBJIEV4Y2VwdGlvbiBoaWVyYXJjaHkgKi8KCiAgICBpZiAoIShweXNxbGl0ZV9FcnJvciA9IFB5RXJyX05ld0V4Y2VwdGlvbihNT0RVTEVfTkFNRSAiLkVycm9yIiwgUHlFeGNfU3RhbmRhcmRFcnJvciwgTlVMTCkpKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJFcnJvciIsIHB5c3FsaXRlX0Vycm9yKTsKCiAgICBpZiAoIShweXNxbGl0ZV9XYXJuaW5nID0gUHlFcnJfTmV3RXhjZXB0aW9uKE1PRFVMRV9OQU1FICIuV2FybmluZyIsIFB5RXhjX1N0YW5kYXJkRXJyb3IsIE5VTEwpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAiV2FybmluZyIsIHB5c3FsaXRlX1dhcm5pbmcpOwoKICAgIC8qIEVycm9yIHN1YmNsYXNzZXMgKi8KCiAgICBpZiAoIShweXNxbGl0ZV9JbnRlcmZhY2VFcnJvciA9IFB5RXJyX05ld0V4Y2VwdGlvbihNT0RVTEVfTkFNRSAiLkludGVyZmFjZUVycm9yIiwgcHlzcWxpdGVfRXJyb3IsIE5VTEwpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAiSW50ZXJmYWNlRXJyb3IiLCBweXNxbGl0ZV9JbnRlcmZhY2VFcnJvcik7CgogICAgaWYgKCEocHlzcWxpdGVfRGF0YWJhc2VFcnJvciA9IFB5RXJyX05ld0V4Y2VwdGlvbihNT0RVTEVfTkFNRSAiLkRhdGFiYXNlRXJyb3IiLCBweXNxbGl0ZV9FcnJvciwgTlVMTCkpKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJEYXRhYmFzZUVycm9yIiwgcHlzcWxpdGVfRGF0YWJhc2VFcnJvcik7CgogICAgLyogcHlzcWxpdGVfRGF0YWJhc2VFcnJvciBzdWJjbGFzc2VzICovCgogICAgaWYgKCEocHlzcWxpdGVfSW50ZXJuYWxFcnJvciA9IFB5RXJyX05ld0V4Y2VwdGlvbihNT0RVTEVfTkFNRSAiLkludGVybmFsRXJyb3IiLCBweXNxbGl0ZV9EYXRhYmFzZUVycm9yLCBOVUxMKSkpIHsKICAgICAgICBnb3RvIGVycm9yOwogICAgfQogICAgUHlEaWN0X1NldEl0ZW1TdHJpbmcoZGljdCwgIkludGVybmFsRXJyb3IiLCBweXNxbGl0ZV9JbnRlcm5hbEVycm9yKTsKCiAgICBpZiAoIShweXNxbGl0ZV9PcGVyYXRpb25hbEVycm9yID0gUHlFcnJfTmV3RXhjZXB0aW9uKE1PRFVMRV9OQU1FICIuT3BlcmF0aW9uYWxFcnJvciIsIHB5c3FsaXRlX0RhdGFiYXNlRXJyb3IsIE5VTEwpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAiT3BlcmF0aW9uYWxFcnJvciIsIHB5c3FsaXRlX09wZXJhdGlvbmFsRXJyb3IpOwoKICAgIGlmICghKHB5c3FsaXRlX1Byb2dyYW1taW5nRXJyb3IgPSBQeUVycl9OZXdFeGNlcHRpb24oTU9EVUxFX05BTUUgIi5Qcm9ncmFtbWluZ0Vycm9yIiwgcHlzcWxpdGVfRGF0YWJhc2VFcnJvciwgTlVMTCkpKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJQcm9ncmFtbWluZ0Vycm9yIiwgcHlzcWxpdGVfUHJvZ3JhbW1pbmdFcnJvcik7CgogICAgaWYgKCEocHlzcWxpdGVfSW50ZWdyaXR5RXJyb3IgPSBQeUVycl9OZXdFeGNlcHRpb24oTU9EVUxFX05BTUUgIi5JbnRlZ3JpdHlFcnJvciIsIHB5c3FsaXRlX0RhdGFiYXNlRXJyb3IsTlVMTCkpKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJJbnRlZ3JpdHlFcnJvciIsIHB5c3FsaXRlX0ludGVncml0eUVycm9yKTsKCiAgICBpZiAoIShweXNxbGl0ZV9EYXRhRXJyb3IgPSBQeUVycl9OZXdFeGNlcHRpb24oTU9EVUxFX05BTUUgIi5EYXRhRXJyb3IiLCBweXNxbGl0ZV9EYXRhYmFzZUVycm9yLCBOVUxMKSkpIHsKICAgICAgICBnb3RvIGVycm9yOwogICAgfQogICAgUHlEaWN0X1NldEl0ZW1TdHJpbmcoZGljdCwgIkRhdGFFcnJvciIsIHB5c3FsaXRlX0RhdGFFcnJvcik7CgogICAgaWYgKCEocHlzcWxpdGVfTm90U3VwcG9ydGVkRXJyb3IgPSBQeUVycl9OZXdFeGNlcHRpb24oTU9EVUxFX05BTUUgIi5Ob3RTdXBwb3J0ZWRFcnJvciIsIHB5c3FsaXRlX0RhdGFiYXNlRXJyb3IsIE5VTEwpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAiTm90U3VwcG9ydGVkRXJyb3IiLCBweXNxbGl0ZV9Ob3RTdXBwb3J0ZWRFcnJvcik7CgogICAgLyogV2UganVzdCBuZWVkICJzb21ldGhpbmciIHVuaXF1ZSBmb3IgcHlzcWxpdGVfT3B0aW1pemVkVW5pY29kZS4gSXQgZG9lcyBub3QgcmVhbGx5CiAgICAgKiBuZWVkIHRvIGJlIGEgc3RyaW5nIHN1YmNsYXNzLiBKdXN0IGFueXRoaW5nIHRoYXQgY2FuIGFjdCBhcyBhIHNwZWNpYWwKICAgICAqIG1hcmtlciBmb3IgdXMuIFNvIEkgcHVsbGVkIFB5Q2VsbF9UeXBlIG91dCBvZiBteSBtYWdpYyBoYXQuCiAgICAgKi8KICAgIFB5X0lOQ1JFRigoUHlPYmplY3QqKSZQeUNlbGxfVHlwZSk7CiAgICBweXNxbGl0ZV9PcHRpbWl6ZWRVbmljb2RlID0gKFB5T2JqZWN0KikmUHlDZWxsX1R5cGU7CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAiT3B0aW1pemVkVW5pY29kZSIsIHB5c3FsaXRlX09wdGltaXplZFVuaWNvZGUpOwoKICAgIC8qIFNldCBpbnRlZ2VyIGNvbnN0YW50cyAqLwogICAgZm9yIChpID0gMDsgX2ludF9jb25zdGFudHNbaV0uY29uc3RhbnRfbmFtZSAhPSAwOyBpKyspIHsKICAgICAgICB0bXBfb2JqID0gUHlJbnRfRnJvbUxvbmcoX2ludF9jb25zdGFudHNbaV0uY29uc3RhbnRfdmFsdWUpOwogICAgICAgIGlmICghdG1wX29iaikgewogICAgICAgICAgICBnb3RvIGVycm9yOwogICAgICAgIH0KICAgICAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCBfaW50X2NvbnN0YW50c1tpXS5jb25zdGFudF9uYW1lLCB0bXBfb2JqKTsKICAgICAgICBQeV9ERUNSRUYodG1wX29iaik7CiAgICB9CgogICAgaWYgKCEodG1wX29iaiA9IFB5U3RyaW5nX0Zyb21TdHJpbmcoUFlTUUxJVEVfVkVSU0lPTikpKSB7CiAgICAgICAgZ290byBlcnJvcjsKICAgIH0KICAgIFB5RGljdF9TZXRJdGVtU3RyaW5nKGRpY3QsICJ2ZXJzaW9uIiwgdG1wX29iaik7CiAgICBQeV9ERUNSRUYodG1wX29iaik7CgogICAgaWYgKCEodG1wX29iaiA9IFB5U3RyaW5nX0Zyb21TdHJpbmcoc3FsaXRlM19saWJ2ZXJzaW9uKCkpKSkgewogICAgICAgIGdvdG8gZXJyb3I7CiAgICB9CiAgICBQeURpY3RfU2V0SXRlbVN0cmluZyhkaWN0LCAic3FsaXRlX3ZlcnNpb24iLCB0bXBfb2JqKTsKICAgIFB5X0RFQ1JFRih0bXBfb2JqKTsKCiAgICAvKiBpbml0aWFsaXplIG1pY3JvcHJvdG9jb2xzIGxheWVyICovCiAgICBtaWNyb3Byb3RvY29sc19pbml0KGRpY3QpOwoKICAgIC8qIGluaXRpYWxpemUgdGhlIGRlZmF1bHQgY29udmVydGVycyAqLwogICAgY29udmVydGVyc19pbml0KGRpY3QpOwoKICAgIF9lbmFibGVfY2FsbGJhY2tfdHJhY2ViYWNrcyA9IDA7CgogICAgcHlzcWxpdGVfQmFzZVR5cGVBZGFwdGVkID0gMDsKCiAgICAvKiBPcmlnaW5hbCBjb21tZW50IGZvcm0gX2JzZGRiLmMgaW4gdGhlIFB5dGhvbiBjb3JlLiBUaGlzIGlzIGFsc28gc3RpbGwKICAgICAqIG5lZWRlZCBub3dhZGF5cyBmb3IgUHl0aG9uIDIuMy8yLjQuCiAgICAgKiAKICAgICAqIFB5RXZhbF9Jbml0VGhyZWFkcyBpcyBjYWxsZWQgaGVyZSBkdWUgdG8gYSBxdWlyayBpbiBweXRob24gMS41CiAgICAgKiAtIDIuMi4xIChhdCBsZWFzdCkgYWNjb3JkaW5nIHRvIFJ1c3NlbGwgV2lsbGlhbXNvbiA8bWVyZWxAd3QubmV0PjoKICAgICAqIFRoZSBnbG9iYWwgaW50ZXJlcHJldGVyIGxvY2sgaXMgbm90IGluaXRpYWxpemVkIHVudGlsIHRoZSBmaXJzdAogICAgICogdGhyZWFkIGlzIGNyZWF0ZWQgdXNpbmcgdGhyZWFkLnN0YXJ0X25ld190aHJlYWQoKSBvciBmb3JrKCkgaXMKICAgICAqIGNhbGxlZC4gIHRoYXQgd291bGQgY2F1c2UgdGhlIEFMTE9XX1RIUkVBRFMgaGVyZSB0byBzZWdmYXVsdCBkdWUKICAgICAqIHRvIGEgbnVsbCBwb2ludGVyIHJlZmVyZW5jZSBpZiBubyB0aHJlYWRzIG9yIGNoaWxkIHByb2Nlc3NlcwogICAgICogaGF2ZSBiZWVuIGNyZWF0ZWQuICBUaGlzIHdvcmtzIGFyb3VuZCB0aGF0IGFuZCBpcyBhIG5vLW9wIGlmCiAgICAgKiB0aHJlYWRzIGhhdmUgYWxyZWFkeSBiZWVuIGluaXRpYWxpemVkLgogICAgICogIChzZWUgcHlic2RkYi11c2VycyBtYWlsaW5nIGxpc3QgcG9zdCBvbiAyMDAyLTA4LTA3KQogICAgICovCiAgICBQeUV2YWxfSW5pdFRocmVhZHMoKTsKCmVycm9yOgogICAgaWYgKFB5RXJyX09jY3VycmVkKCkpCiAgICB7CiAgICAgICAgUHlFcnJfU2V0U3RyaW5nKFB5RXhjX0ltcG9ydEVycm9yLCBNT0RVTEVfTkFNRSAiOiBpbml0IGZhaWxlZCIpOwogICAgfQp9Cg==