LyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiAgIHJwbmcgLSBzaW1wbGUgUE5HIGRpc3BsYXkgcHJvZ3JhbSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBycG5nLXguYwoKICAgVGhpcyBwcm9ncmFtIGRlY29kZXMgYW5kIGRpc3BsYXlzIFBORyBpbWFnZXMsIHdpdGggZ2FtbWEgY29ycmVjdGlvbiBhbmQKICAgb3B0aW9uYWxseSB3aXRoIGEgdXNlci1zcGVjaWZpZWQgYmFja2dyb3VuZCBjb2xvciAoaW4gY2FzZSB0aGUgaW1hZ2UgaGFzCiAgIHRyYW5zcGFyZW5jeSkuICBJdCBpcyB2ZXJ5IG5lYXJseSB0aGUgbW9zdCBiYXNpYyBQTkcgdmlld2VyIHBvc3NpYmxlLgogICBUaGlzIHZlcnNpb24gaXMgZm9yIHRoZSBYIFdpbmRvdyBTeXN0ZW0gKHRlc3RlZCBieSBhdXRob3IgdW5kZXIgVW5peCBhbmQKICAgYnkgTWFydGluIFppbnNlciB1bmRlciBPcGVuVk1TOyBtYXkgd29yayB1bmRlciBPUy8yIHdpdGggc29tZSB0d2Vha2luZykuCgogICB0byBkbzoKICAgIC0gOC1iaXQgKGNvbG9ybWFwcGVkKSBYIHN1cHBvcnQKICAgIC0gdXNlICUuMTAyM3MgdG8gc2ltcGxpZnkgdHJ1bmNhdGlvbiBvZiB0aXRsZS1iYXIgc3RyaW5nPwoKICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiAgIENoYW5nZWxvZzoKICAgIC0gMS4wMTogIGluaXRpYWwgcHVibGljIHJlbGVhc2UKICAgIC0gMS4wMjogIG1vZGlmaWVkIHRvIGFsbG93IGFiYnJldmlhdGVkIG9wdGlvbnM7IGZpeGVkIGxvbmcvdWxvbmcgbWlzLQogICAgICAgICAgICAgIG1hdGNoOyBzd2l0Y2hlZCB0byBwbmdfam1wYnVmKCkgbWFjcm8KICAgIC0gMS4xMDogIGFkZGVkIHN1cHBvcnQgZm9yIG5vbi1kZWZhdWx0IHZpc3VhbHM7IGZpeGVkIFggcGl4ZWwtY29udmVyc2lvbgogICAgLSAxLjExOiAgYWRkZWQgZXh0cmEgc2V0IG9mIHBhcmVudGhlc2VzIHRvIHBuZ19qbXBidWYoKSBtYWNybzsgZml4ZWQKICAgICAgICAgICAgICBjb21tYW5kLWxpbmUgcGFyc2luZyBidWcKICAgIC0gMS4xMjogIGZpeGVkIHNvbWUgc21hbGwgWCBtZW1vcnkgbGVha3MgKHRoYW5rcyB0byBGcmFu529pcyBQZXRpdGplYW4pCiAgICAtIDEuMTM6ICBmaXhlZCBYRnJlZUdDKCkgY3Jhc2ggYnVnICh0aGFua3MgdG8gUGF0cmljayBXZWxjaGUpCiAgICAtIDEuMTQ6ICBhZGRlZCBzdXBwb3J0IGZvciBYIHJlc291cmNlcyAodGhhbmtzIHRvIEdlcmhhcmQgTmlrbGFzY2gpCiAgICAtIDIuMDA6ICBkdWFsLWxpY2Vuc2VkIChhZGRlZCBHTlUgR1BMKQogICAgLSAyLjAxOiAgZml4ZWQgaW1wcm9wZXIgZGlzcGxheSBvZiB1c2FnZSBzY3JlZW4gb24gUE5HIGVycm9yKHMpCgogIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKICAgICAgQ29weXJpZ2h0IChjKSAxOTk4LTIwMDggR3JlZyBSb2Vsb2ZzLiAgQWxsIHJpZ2h0cyByZXNlcnZlZC4KCiAgICAgIFRoaXMgc29mdHdhcmUgaXMgcHJvdmlkZWQgImFzIGlzLCIgd2l0aG91dCB3YXJyYW50eSBvZiBhbnkga2luZCwKICAgICAgZXhwcmVzcyBvciBpbXBsaWVkLiAgSW4gbm8gZXZlbnQgc2hhbGwgdGhlIGF1dGhvciBvciBjb250cmlidXRvcnMKICAgICAgYmUgaGVsZCBsaWFibGUgZm9yIGFueSBkYW1hZ2VzIGFyaXNpbmcgaW4gYW55IHdheSBmcm9tIHRoZSB1c2Ugb2YKICAgICAgdGhpcyBzb2Z0d2FyZS4KCiAgICAgIFRoZSBjb250ZW50cyBvZiB0aGlzIGZpbGUgYXJlIERVQUwtTElDRU5TRUQuICBZb3UgbWF5IG1vZGlmeSBhbmQvb3IKICAgICAgcmVkaXN0cmlidXRlIHRoaXMgc29mdHdhcmUgYWNjb3JkaW5nIHRvIHRoZSB0ZXJtcyBvZiBvbmUgb2YgdGhlCiAgICAgIGZvbGxvd2luZyB0d28gbGljZW5zZXMgKGF0IHlvdXIgb3B0aW9uKToKCgogICAgICBMSUNFTlNFIDEgKCJCU0QtbGlrZSB3aXRoIGFkdmVydGlzaW5nIGNsYXVzZSIpOgoKICAgICAgUGVybWlzc2lvbiBpcyBncmFudGVkIHRvIGFueW9uZSB0byB1c2UgdGhpcyBzb2Z0d2FyZSBmb3IgYW55IHB1cnBvc2UsCiAgICAgIGluY2x1ZGluZyBjb21tZXJjaWFsIGFwcGxpY2F0aW9ucywgYW5kIHRvIGFsdGVyIGl0IGFuZCByZWRpc3RyaWJ1dGUKICAgICAgaXQgZnJlZWx5LCBzdWJqZWN0IHRvIHRoZSBmb2xsb3dpbmcgcmVzdHJpY3Rpb25zOgoKICAgICAgMS4gUmVkaXN0cmlidXRpb25zIG9mIHNvdXJjZSBjb2RlIG11c3QgcmV0YWluIHRoZSBhYm92ZSBjb3B5cmlnaHQKICAgICAgICAgbm90aWNlLCBkaXNjbGFpbWVyLCBhbmQgdGhpcyBsaXN0IG9mIGNvbmRpdGlvbnMuCiAgICAgIDIuIFJlZGlzdHJpYnV0aW9ucyBpbiBiaW5hcnkgZm9ybSBtdXN0IHJlcHJvZHVjZSB0aGUgYWJvdmUgY29weXJpZ2h0CiAgICAgICAgIG5vdGljZSwgZGlzY2xhaW1lciwgYW5kIHRoaXMgbGlzdCBvZiBjb25kaXRpb25zIGluIHRoZSBkb2N1bWVudGEtCiAgICAgICAgIHRpb24gYW5kL29yIG90aGVyIG1hdGVyaWFscyBwcm92aWRlZCB3aXRoIHRoZSBkaXN0cmlidXRpb24uCiAgICAgIDMuIEFsbCBhZHZlcnRpc2luZyBtYXRlcmlhbHMgbWVudGlvbmluZyBmZWF0dXJlcyBvciB1c2Ugb2YgdGhpcwogICAgICAgICBzb2Z0d2FyZSBtdXN0IGRpc3BsYXkgdGhlIGZvbGxvd2luZyBhY2tub3dsZWRnbWVudDoKCiAgICAgICAgICAgIFRoaXMgcHJvZHVjdCBpbmNsdWRlcyBzb2Z0d2FyZSBkZXZlbG9wZWQgYnkgR3JlZyBSb2Vsb2ZzCiAgICAgICAgICAgIGFuZCBjb250cmlidXRvcnMgZm9yIHRoZSBib29rLCAiUE5HOiBUaGUgRGVmaW5pdGl2ZSBHdWlkZSwiCiAgICAgICAgICAgIHB1Ymxpc2hlZCBieSBPJ1JlaWxseSBhbmQgQXNzb2NpYXRlcy4KCgogICAgICBMSUNFTlNFIDIgKEdOVSBHUEwgdjIgb3IgbGF0ZXIpOgoKICAgICAgVGhpcyBwcm9ncmFtIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkKICAgICAgaXQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZSBhcyBwdWJsaXNoZWQgYnkKICAgICAgdGhlIEZyZWUgU29mdHdhcmUgRm91bmRhdGlvbjsgZWl0aGVyIHZlcnNpb24gMiBvZiB0aGUgTGljZW5zZSwgb3IKICAgICAgKGF0IHlvdXIgb3B0aW9uKSBhbnkgbGF0ZXIgdmVyc2lvbi4KCiAgICAgIFRoaXMgcHJvZ3JhbSBpcyBkaXN0cmlidXRlZCBpbiB0aGUgaG9wZSB0aGF0IGl0IHdpbGwgYmUgdXNlZnVsLAogICAgICBidXQgV0lUSE9VVCBBTlkgV0FSUkFOVFk7IHdpdGhvdXQgZXZlbiB0aGUgaW1wbGllZCB3YXJyYW50eSBvZgogICAgICBNRVJDSEFOVEFCSUxJVFkgb3IgRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UuICBTZWUgdGhlCiAgICAgIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGZvciBtb3JlIGRldGFpbHMuCgogICAgICBZb3Ugc2hvdWxkIGhhdmUgcmVjZWl2ZWQgYSBjb3B5IG9mIHRoZSBHTlUgR2VuZXJhbCBQdWJsaWMgTGljZW5zZQogICAgICBhbG9uZyB3aXRoIHRoaXMgcHJvZ3JhbTsgaWYgbm90LCB3cml0ZSB0byB0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLAogICAgICBJbmMuLCA1OSBUZW1wbGUgUGxhY2UsIFN1aXRlIDMzMCwgQm9zdG9uLCBNQSAgMDIxMTEtMTMwNyAgVVNBCgogIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovCgojZGVmaW5lIFBST0dOQU1FICAicnBuZy14IgojZGVmaW5lIExPTkdOQU1FICAiU2ltcGxlIFBORyBWaWV3ZXIgZm9yIFgiCiNkZWZpbmUgVkVSU0lPTiAgICIyLjAxIG9mIDE2IE1hcmNoIDIwMDgiCiNkZWZpbmUgUkVTTkFNRSAgICJycG5nIgkvKiBvdXIgWCByZXNvdXJjZSBhcHBsaWNhdGlvbiBuYW1lICovCiNkZWZpbmUgUkVTQ0xBU1MgICJScG5nIgkvKiBvdXIgWCByZXNvdXJjZSBjbGFzcyBuYW1lICovCgojaW5jbHVkZSA8c3RkaW8uaD4KI2luY2x1ZGUgPHN0ZGxpYi5oPgojaW5jbHVkZSA8c3RyaW5nLmg+CiNpbmNsdWRlIDx0aW1lLmg+CiNpbmNsdWRlIDxYMTEvWGxpYi5oPgojaW5jbHVkZSA8WDExL1h1dGlsLmg+CiNpbmNsdWRlIDxYMTEvWG9zLmg+CiNpbmNsdWRlIDxYMTEva2V5c3ltLmg+CgovKiAjZGVmaW5lIERFQlVHICA6ICB0aGlzIGVuYWJsZXMgdGhlIFRyYWNlKCkgbWFjcm9zICovCgojaW5jbHVkZSAicmVhZHBuZy5oIiAgIC8qIHR5cGVkZWZzLCBjb21tb24gbWFjcm9zLCByZWFkcG5nIHByb3RvdHlwZXMgKi8KCgovKiBjb3VsZCBqdXN0IGluY2x1ZGUgcG5nLmgsIGJ1dCB0aGlzIG1hY3JvIGlzIHRoZSBvbmx5IHRoaW5nIHdlIG5lZWQKICogKG5hbWUgYW5kIHR5cGVkZWZzIGNoYW5nZWQgdG8gbG9jYWwgdmVyc2lvbnMpOyBub3RlIHRoYXQgc2lkZSBlZmZlY3RzCiAqIG9ubHkgaGFwcGVuIHdpdGggYWxwaGEgKHdoaWNoIGNvdWxkIGVhc2lseSBiZSBhdm9pZGVkIHdpdGgKICogInVzaCBhY29weSA9IChhbHBoYSk7IikgKi8KCiNkZWZpbmUgYWxwaGFfY29tcG9zaXRlKGNvbXBvc2l0ZSwgZmcsIGFscGhhLCBiZykgeyAgICAgICAgICAgICAgIFwKICAgIHVzaCB0ZW1wID0gKCh1c2gpKGZnKSoodXNoKShhbHBoYSkgKyAgICAgICAgICAgICAgICAgICAgICAgICAgXAogICAgICAgICAgICAgICAgKHVzaCkoYmcpKih1c2gpKDI1NSAtICh1c2gpKGFscGhhKSkgKyAodXNoKTEyOCk7ICBcCiAgICAoY29tcG9zaXRlKSA9ICh1Y2gpKCh0ZW1wICsgKHRlbXAgPj4gOCkpID4+IDgpOyAgICAgICAgICAgICAgIFwKfQoKCi8qIGxvY2FsIHByb3RvdHlwZXMgKi8Kc3RhdGljIGludCAgcnBuZ194X2NyZWF0ZV93aW5kb3codm9pZCk7CnN0YXRpYyBpbnQgIHJwbmdfeF9kaXNwbGF5X2ltYWdlKHZvaWQpOwpzdGF0aWMgdm9pZCBycG5nX3hfY2xlYW51cCh2b2lkKTsKc3RhdGljIGludCAgcnBuZ194X21zYih1bGcgdTMydmFsKTsKCgpzdGF0aWMgY2hhciB0aXRsZWJhclsxMDI0XSwgKndpbmRvd19uYW1lID0gdGl0bGViYXI7CnN0YXRpYyBjaGFyICphcHBuYW1lID0gTE9OR05BTUU7CnN0YXRpYyBjaGFyICppY29uX25hbWUgPSBQUk9HTkFNRTsKc3RhdGljIGNoYXIgKnJlc19uYW1lID0gUkVTTkFNRTsKc3RhdGljIGNoYXIgKnJlc19jbGFzcyA9IFJFU0NMQVNTOwpzdGF0aWMgY2hhciAqZmlsZW5hbWU7CnN0YXRpYyBGSUxFICppbmZpbGU7CgpzdGF0aWMgY2hhciAqYmdzdHI7CnN0YXRpYyB1Y2ggYmdfcmVkPTAsIGJnX2dyZWVuPTAsIGJnX2JsdWU9MDsKCnN0YXRpYyBkb3VibGUgZGlzcGxheV9leHBvbmVudDsKCnN0YXRpYyB1bGcgaW1hZ2Vfd2lkdGgsIGltYWdlX2hlaWdodCwgaW1hZ2Vfcm93Ynl0ZXM7CnN0YXRpYyBpbnQgaW1hZ2VfY2hhbm5lbHM7CnN0YXRpYyB1Y2ggKmltYWdlX2RhdGE7CgovKiBYLXNwZWNpZmljIHZhcmlhYmxlcyAqLwpzdGF0aWMgY2hhciAqZGlzcGxheW5hbWU7CnN0YXRpYyBYSW1hZ2UgKnhpbWFnZTsKc3RhdGljIERpc3BsYXkgKmRpc3BsYXk7CnN0YXRpYyBpbnQgZGVwdGg7CnN0YXRpYyBWaXN1YWwgKnZpc3VhbDsKc3RhdGljIFhWaXN1YWxJbmZvICp2aXN1YWxfbGlzdDsKc3RhdGljIGludCBSU2hpZnQsIEdTaGlmdCwgQlNoaWZ0OwpzdGF0aWMgdWxnIFJNYXNrLCBHTWFzaywgQk1hc2s7CnN0YXRpYyBXaW5kb3cgd2luZG93OwpzdGF0aWMgR0MgZ2M7CnN0YXRpYyBDb2xvcm1hcCBjb2xvcm1hcDsKCnN0YXRpYyBpbnQgaGF2ZV9ub25kZWZhdWx0X3Zpc3VhbCA9IEZBTFNFOwpzdGF0aWMgaW50IGhhdmVfY29sb3JtYXAgPSBGQUxTRTsKc3RhdGljIGludCBoYXZlX3dpbmRvdyA9IEZBTFNFOwpzdGF0aWMgaW50IGhhdmVfZ2MgPSBGQUxTRTsKLyoKdWxnIG51bWNvbG9ycz0wLCBwaXhlbHNbMjU2XTsKdXNoIHJlZHNbMjU2XSwgZ3JlZW5zWzI1Nl0sIGJsdWVzWzI1Nl07CiAqLwoKCgoKaW50IG1haW4oaW50IGFyZ2MsIGNoYXIgKiphcmd2KQp7CiNpZmRlZiBzZ2kKICAgIGNoYXIgdG1wbGluZVs4MF07CiNlbmRpZgogICAgY2hhciAqcDsKICAgIGludCByYywgYWxlbiwgZmxlbjsKICAgIGludCBlcnJvciA9IDA7CiAgICBpbnQgaGF2ZV9iZyA9IEZBTFNFOwogICAgZG91YmxlIExVVF9leHBvbmVudDsgICAgICAgICAgICAgICAvKiBqdXN0IHRoZSBsb29rdXAgdGFibGUgKi8KICAgIGRvdWJsZSBDUlRfZXhwb25lbnQgPSAyLjI7ICAgICAgICAgLyoganVzdCB0aGUgbW9uaXRvciAqLwogICAgZG91YmxlIGRlZmF1bHRfZGlzcGxheV9leHBvbmVudDsgICAvKiB3aG9sZSBkaXNwbGF5IHN5c3RlbSAqLwogICAgWEV2ZW50IGU7CiAgICBLZXlTeW0gazsKCgogICAgZGlzcGxheW5hbWUgPSAoY2hhciAqKU5VTEw7CiAgICBmaWxlbmFtZSA9IChjaGFyICopTlVMTDsKCgogICAgLyogRmlyc3Qgc2V0IHRoZSBkZWZhdWx0IHZhbHVlIGZvciBvdXIgZGlzcGxheS1zeXN0ZW0gZXhwb25lbnQsIGkuZS4sCiAgICAgKiB0aGUgcHJvZHVjdCBvZiB0aGUgQ1JUIGV4cG9uZW50IGFuZCB0aGUgZXhwb25lbnQgY29ycmVzcG9uZGluZyB0bwogICAgICogdGhlIGZyYW1lLWJ1ZmZlcidzIGxvb2t1cCB0YWJsZSAoTFVUKSwgaWYgYW55LiAgVGhpcyBpcyBub3QgYW4KICAgICAqIGV4aGF1c3RpdmUgbGlzdCBvZiBMVVQgdmFsdWVzIChlLmcuLCBPcGVuU3RlcCBoYXMgYSBsb3Qgb2Ygd2VpcmQKICAgICAqIG9uZXMpLCBidXQgaXQgc2hvdWxkIGNvdmVyIDk5JSBvZiB0aGUgY3VycmVudCBwb3NzaWJpbGl0aWVzLiAqLwoKI2lmIGRlZmluZWQoTmVYVCkKICAgIExVVF9leHBvbmVudCA9IDEuMCAvIDIuMjsKICAgIC8qCiAgICBpZiAoc29tZV9uZXh0X2Z1bmN0aW9uX3RoYXRfcmV0dXJuc19nYW1tYSgmbmV4dF9nYW1tYSkpCiAgICAgICAgTFVUX2V4cG9uZW50ID0gMS4wIC8gbmV4dF9nYW1tYTsKICAgICAqLwojZWxpZiBkZWZpbmVkKHNnaSkKICAgIExVVF9leHBvbmVudCA9IDEuMCAvIDEuNzsKICAgIC8qIHRoZXJlIGRvZXNuJ3Qgc2VlbSB0byBiZSBhbnkgZG9jdW1lbnRlZCBmdW5jdGlvbiB0byBnZXQgdGhlCiAgICAgKiAiZ2FtbWEiIHZhbHVlLCBzbyB3ZSBkbyBpdCB0aGUgaGFyZCB3YXkgKi8KICAgIGluZmlsZSA9IGZvcGVuKCIvZXRjL2NvbmZpZy9zeXN0ZW0uZ2xHYW1tYVZhbCIsICJyIik7CiAgICBpZiAoaW5maWxlKSB7CiAgICAgICAgZG91YmxlIHNnaV9nYW1tYTsKCiAgICAgICAgZmdldHModG1wbGluZSwgODAsIGluZmlsZSk7CiAgICAgICAgZmNsb3NlKGluZmlsZSk7CiAgICAgICAgc2dpX2dhbW1hID0gYXRvZih0bXBsaW5lKTsKICAgICAgICBpZiAoc2dpX2dhbW1hID4gMC4wKQogICAgICAgICAgICBMVVRfZXhwb25lbnQgPSAxLjAgLyBzZ2lfZ2FtbWE7CiAgICB9CiNlbGlmIGRlZmluZWQoTWFjaW50b3NoKQogICAgTFVUX2V4cG9uZW50ID0gMS44IC8gMi42MTsKICAgIC8qCiAgICBpZiAoc29tZV9tYWNfZnVuY3Rpb25fdGhhdF9yZXR1cm5zX2dhbW1hKCZtYWNfZ2FtbWEpKQogICAgICAgIExVVF9leHBvbmVudCA9IG1hY19nYW1tYSAvIDIuNjE7CiAgICAgKi8KI2Vsc2UKICAgIExVVF9leHBvbmVudCA9IDEuMDsgICAvKiBhc3N1bWUgbm8gTFVUOiAgbW9zdCBQQ3MgKi8KI2VuZGlmCgogICAgLyogdGhlIGRlZmF1bHRzIGFib3ZlIGdpdmUgMS4wLCAxLjMsIDEuNSBhbmQgMi4yLCByZXNwZWN0aXZlbHk6ICovCiAgICBkZWZhdWx0X2Rpc3BsYXlfZXhwb25lbnQgPSBMVVRfZXhwb25lbnQgKiBDUlRfZXhwb25lbnQ7CgoKICAgIC8qIElmIHRoZSB1c2VyIGhhcyBzZXQgdGhlIFNDUkVFTl9HQU1NQSBlbnZpcm9ubWVudCB2YXJpYWJsZSBhcyBzdWdnZXN0ZWQKICAgICAqIChzb21ld2hhdCBpbXByZWNpc2VseSkgaW4gdGhlIGxpYnBuZyBkb2N1bWVudGF0aW9uLCB1c2UgdGhhdDsgb3RoZXJ3aXNlCiAgICAgKiB1c2UgdGhlIGRlZmF1bHQgdmFsdWUgd2UganVzdCBjYWxjdWxhdGVkLiAgRWl0aGVyIHdheSwgdGhlIHVzZXIgbWF5CiAgICAgKiBvdmVycmlkZSB0aGlzIHZpYSBhIGNvbW1hbmQtbGluZSBvcHRpb24uICovCgogICAgaWYgKChwID0gZ2V0ZW52KCJTQ1JFRU5fR0FNTUEiKSkgIT0gTlVMTCkKICAgICAgICBkaXNwbGF5X2V4cG9uZW50ID0gYXRvZihwKTsKICAgIGVsc2UKICAgICAgICBkaXNwbGF5X2V4cG9uZW50ID0gZGVmYXVsdF9kaXNwbGF5X2V4cG9uZW50OwoKCiAgICAvKiBOb3cgcGFyc2UgdGhlIGNvbW1hbmQgbGluZSBmb3Igb3B0aW9ucyBhbmQgdGhlIFBORyBmaWxlbmFtZS4gKi8KCiAgICB3aGlsZSAoKisrYXJndiAmJiAhZXJyb3IpIHsKICAgICAgICBpZiAoIXN0cm5jbXAoKmFyZ3YsICItZGlzcGxheSIsIDIpKSB7CiAgICAgICAgICAgIGlmICghKisrYXJndikKICAgICAgICAgICAgICAgICsrZXJyb3I7CiAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgIGRpc3BsYXluYW1lID0gKmFyZ3Y7CiAgICAgICAgfSBlbHNlIGlmICghc3RybmNtcCgqYXJndiwgIi1nYW1tYSIsIDIpKSB7CiAgICAgICAgICAgIGlmICghKisrYXJndikKICAgICAgICAgICAgICAgICsrZXJyb3I7CiAgICAgICAgICAgIGVsc2UgewogICAgICAgICAgICAgICAgZGlzcGxheV9leHBvbmVudCA9IGF0b2YoKmFyZ3YpOwogICAgICAgICAgICAgICAgaWYgKGRpc3BsYXlfZXhwb25lbnQgPD0gMC4wKQogICAgICAgICAgICAgICAgICAgICsrZXJyb3I7CiAgICAgICAgICAgIH0KICAgICAgICB9IGVsc2UgaWYgKCFzdHJuY21wKCphcmd2LCAiLWJnY29sb3IiLCAyKSkgewogICAgICAgICAgICBpZiAoISorK2FyZ3YpCiAgICAgICAgICAgICAgICArK2Vycm9yOwogICAgICAgICAgICBlbHNlIHsKICAgICAgICAgICAgICAgIGJnc3RyID0gKmFyZ3Y7CiAgICAgICAgICAgICAgICBpZiAoc3RybGVuKGJnc3RyKSAhPSA3IHx8IGJnc3RyWzBdICE9ICcjJykKICAgICAgICAgICAgICAgICAgICArK2Vycm9yOyAKICAgICAgICAgICAgICAgIGVsc2UgCiAgICAgICAgICAgICAgICAgICAgaGF2ZV9iZyA9IFRSVUU7CiAgICAgICAgICAgIH0KICAgICAgICB9IGVsc2UgewogICAgICAgICAgICBpZiAoKiphcmd2ICE9ICctJykgewogICAgICAgICAgICAgICAgZmlsZW5hbWUgPSAqYXJndjsKICAgICAgICAgICAgICAgIGlmIChhcmd2WzFdKSAgIC8qIHNob3VsZG4ndCBiZSBhbnkgbW9yZSBhcmdzIGFmdGVyIGZpbGVuYW1lICovCiAgICAgICAgICAgICAgICAgICAgKytlcnJvcjsKICAgICAgICAgICAgfSBlbHNlCiAgICAgICAgICAgICAgICArK2Vycm9yOyAgIC8qIG5vdCBleHBlY3RpbmcgYW55IG90aGVyIG9wdGlvbnMgKi8KICAgICAgICB9CiAgICB9CgogICAgaWYgKCFmaWxlbmFtZSkKICAgICAgICArK2Vycm9yOwoKCiAgICAvKiBwcmludCB1c2FnZSBzY3JlZW4gaWYgYW55IGVycm9ycyB1cCB0byB0aGlzIHBvaW50ICovCgogICAgaWYgKGVycm9yKSB7CiAgICAgICAgZnByaW50ZihzdGRlcnIsICJcbiVzICVzOiAgJXNcbiIsIFBST0dOQU1FLCBWRVJTSU9OLCBhcHBuYW1lKTsKICAgICAgICByZWFkcG5nX3ZlcnNpb25faW5mbygpOwogICAgICAgIGZwcmludGYoc3RkZXJyLCAiXG4iCiAgICAgICAgICAiVXNhZ2U6ICAlcyBbLWRpc3BsYXkgeGRweV0gWy1nYW1tYSBleHBdIFstYmdjb2xvciBiZ10gZmlsZS5wbmdcbiIKICAgICAgICAgICIgICAgeGRweVx0bmFtZSBvZiB0aGUgdGFyZ2V0IFggZGlzcGxheSAoZS5nLiwgYGBob3N0bmFtZTowJycpXG4iCiAgICAgICAgICAiICAgIGV4cCBcdHRyYW5zZmVyLWZ1bmN0aW9uIGV4cG9uZW50IChgYGdhbW1hJycpIG9mIHRoZSBkaXNwbGF5XG4iCiAgICAgICAgICAiXHRcdCAgc3lzdGVtIGluIGZsb2F0aW5nLXBvaW50IGZvcm1hdCAoZS5nLiwgYGAlLjFmJycpOyBlcXVhbFxuIgogICAgICAgICAgIlx0XHQgIHRvIHRoZSBwcm9kdWN0IG9mIHRoZSBsb29rdXAtdGFibGUgZXhwb25lbnQgKHZhcmllcylcbiIKICAgICAgICAgICJcdFx0ICBhbmQgdGhlIENSVCBleHBvbmVudCAodXN1YWxseSAyLjIpOyBtdXN0IGJlIHBvc2l0aXZlXG4iCiAgICAgICAgICAiICAgIGJnICBcdGRlc2lyZWQgYmFja2dyb3VuZCBjb2xvciBpbiA3LWNoYXJhY3RlciBoZXggUkdCIGZvcm1hdFxuIgogICAgICAgICAgIlx0XHQgIChlLmcuLCBgYCNmZjc3MDAnJyBmb3Igb3JhbmdlOiAgc2FtZSBhcyBIVE1MIGNvbG9ycyk7XG4iCiAgICAgICAgICAiXHRcdCAgdXNlZCB3aXRoIHRyYW5zcGFyZW50IGltYWdlc1xuIgogICAgICAgICAgIlxuUHJlc3MgUSwgRXNjIG9yIG1vdXNlIGJ1dHRvbiAxICh3aXRoaW4gaW1hZ2Ugd2luZG93LCBhZnRlciBpbWFnZVxuIgogICAgICAgICAgImlzIGRpc3BsYXllZCkgdG8gcXVpdC5cbiIKICAgICAgICAgICJcbiIsIFBST0dOQU1FLCBkZWZhdWx0X2Rpc3BsYXlfZXhwb25lbnQpOwogICAgICAgIGV4aXQoMSk7CiAgICB9CgoKICAgIGlmICghKGluZmlsZSA9IGZvcGVuKGZpbGVuYW1lLCAicmIiKSkpIHsKICAgICAgICBmcHJpbnRmKHN0ZGVyciwgUFJPR05BTUUgIjogIGNhbid0IG9wZW4gUE5HIGZpbGUgWyVzXVxuIiwgZmlsZW5hbWUpOwogICAgICAgICsrZXJyb3I7CiAgICB9IGVsc2UgewogICAgICAgIGlmICgocmMgPSByZWFkcG5nX2luaXQoaW5maWxlLCAmaW1hZ2Vfd2lkdGgsICZpbWFnZV9oZWlnaHQpKSAhPSAwKSB7CiAgICAgICAgICAgIHN3aXRjaCAocmMpIHsKICAgICAgICAgICAgICAgIGNhc2UgMToKICAgICAgICAgICAgICAgICAgICBmcHJpbnRmKHN0ZGVyciwgUFJPR05BTUUKICAgICAgICAgICAgICAgICAgICAgICI6ICBbJXNdIGlzIG5vdCBhIFBORyBmaWxlOiBpbmNvcnJlY3Qgc2lnbmF0dXJlXG4iLAogICAgICAgICAgICAgICAgICAgICAgZmlsZW5hbWUpOwogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgY2FzZSAyOgogICAgICAgICAgICAgICAgICAgIGZwcmludGYoc3RkZXJyLCBQUk9HTkFNRQogICAgICAgICAgICAgICAgICAgICAgIjogIFslc10gaGFzIGJhZCBJSERSIChsaWJwbmcgbG9uZ2ptcClcbiIsIGZpbGVuYW1lKTsKICAgICAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgICAgIGNhc2UgNDoKICAgICAgICAgICAgICAgICAgICBmcHJpbnRmKHN0ZGVyciwgUFJPR05BTUUgIjogIGluc3VmZmljaWVudCBtZW1vcnlcbiIpOwogICAgICAgICAgICAgICAgICAgIGJyZWFrOwogICAgICAgICAgICAgICAgZGVmYXVsdDoKICAgICAgICAgICAgICAgICAgICBmcHJpbnRmKHN0ZGVyciwgUFJPR05BTUUKICAgICAgICAgICAgICAgICAgICAgICI6ICB1bmtub3duIHJlYWRwbmdfaW5pdCgpIGVycm9yXG4iKTsKICAgICAgICAgICAgICAgICAgICBicmVhazsKICAgICAgICAgICAgfQogICAgICAgICAgICArK2Vycm9yOwogICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgIGRpc3BsYXkgPSBYT3BlbkRpc3BsYXkoZGlzcGxheW5hbWUpOwogICAgICAgICAgICBpZiAoIWRpc3BsYXkpIHsKICAgICAgICAgICAgICAgIHJlYWRwbmdfY2xlYW51cChUUlVFKTsKICAgICAgICAgICAgICAgIGZwcmludGYoc3RkZXJyLCBQUk9HTkFNRSAiOiAgY2FuJ3Qgb3BlbiBYIGRpc3BsYXkgWyVzXVxuIiwKICAgICAgICAgICAgICAgICAgZGlzcGxheW5hbWU/IGRpc3BsYXluYW1lIDogImRlZmF1bHQiKTsKICAgICAgICAgICAgICAgICsrZXJyb3I7CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICAgICAgaWYgKGVycm9yKQogICAgICAgICAgICBmY2xvc2UoaW5maWxlKTsKICAgIH0KCgogICAgaWYgKGVycm9yKSB7CiAgICAgICAgZnByaW50ZihzdGRlcnIsIFBST0dOQU1FICI6ICBhYm9ydGluZy5cbiIpOwogICAgICAgIGV4aXQoMik7CiAgICB9CgoKICAgIC8qIHNldCB0aGUgdGl0bGUtYmFyIHN0cmluZywgYnV0IG1ha2Ugc3VyZSBidWZmZXIgZG9lc24ndCBvdmVyZmxvdyAqLwoKICAgIGFsZW4gPSBzdHJsZW4oYXBwbmFtZSk7CiAgICBmbGVuID0gc3RybGVuKGZpbGVuYW1lKTsKICAgIGlmIChhbGVuICsgZmxlbiArIDMgPiAxMDIzKQogICAgICAgIHNwcmludGYodGl0bGViYXIsICIlczogIC4uLiVzIiwgYXBwbmFtZSwgZmlsZW5hbWUrKGFsZW4rZmxlbis2LTEwMjMpKTsKICAgIGVsc2UKICAgICAgICBzcHJpbnRmKHRpdGxlYmFyLCAiJXM6ICAlcyIsIGFwcG5hbWUsIGZpbGVuYW1lKTsKCgogICAgLyogaWYgdGhlIHVzZXIgZGlkbid0IHNwZWNpZnkgYSBiYWNrZ3JvdW5kIGNvbG9yIG9uIHRoZSBjb21tYW5kIGxpbmUsCiAgICAgKiBjaGVjayBmb3Igb25lIGluIHRoZSBQTkcgZmlsZS0taWYgbm90LCB0aGUgaW5pdGlhbGl6ZWQgdmFsdWVzIG9mIDAKICAgICAqIChibGFjaykgd2lsbCBiZSB1c2VkICovCgogICAgaWYgKGhhdmVfYmcpIHsKICAgICAgICB1bnNpZ25lZCByLCBnLCBiOyAgIC8qIHRoaXMgYXBwcm9hY2ggcXVpZXRzIGNvbXBpbGVyIHdhcm5pbmdzICovCgogICAgICAgIHNzY2FuZihiZ3N0cisxLCAiJTJ4JTJ4JTJ4IiwgJnIsICZnLCAmYik7CiAgICAgICAgYmdfcmVkICAgPSAodWNoKXI7CiAgICAgICAgYmdfZ3JlZW4gPSAodWNoKWc7CiAgICAgICAgYmdfYmx1ZSAgPSAodWNoKWI7CiAgICB9IGVsc2UgaWYgKHJlYWRwbmdfZ2V0X2JnY29sb3IoJmJnX3JlZCwgJmJnX2dyZWVuLCAmYmdfYmx1ZSkgPiAxKSB7CiAgICAgICAgcmVhZHBuZ19jbGVhbnVwKFRSVUUpOwogICAgICAgIGZwcmludGYoc3RkZXJyLCBQUk9HTkFNRQogICAgICAgICAgIjogIGxpYnBuZyBlcnJvciB3aGlsZSBjaGVja2luZyBmb3IgYmFja2dyb3VuZCBjb2xvclxuIik7CiAgICAgICAgZXhpdCgyKTsKICAgIH0KCgogICAgLyogZG8gdGhlIGJhc2ljIFggaW5pdGlhbGl6YXRpb24gc3R1ZmYsIG1ha2UgdGhlIHdpbmRvdyBhbmQgZmlsbCBpdAogICAgICogd2l0aCB0aGUgYmFja2dyb3VuZCBjb2xvciAqLwoKICAgIGlmIChycG5nX3hfY3JlYXRlX3dpbmRvdygpKQogICAgICAgIGV4aXQoMik7CgoKICAgIC8qIGRlY29kZSB0aGUgaW1hZ2UsIGFsbCBhdCBvbmNlICovCgogICAgVHJhY2UoKHN0ZGVyciwgImNhbGxpbmcgcmVhZHBuZ19nZXRfaW1hZ2UoKVxuIikpCiAgICBpbWFnZV9kYXRhID0gcmVhZHBuZ19nZXRfaW1hZ2UoZGlzcGxheV9leHBvbmVudCwgJmltYWdlX2NoYW5uZWxzLAogICAgICAmaW1hZ2Vfcm93Ynl0ZXMpOwogICAgVHJhY2UoKHN0ZGVyciwgImRvbmUgd2l0aCByZWFkcG5nX2dldF9pbWFnZSgpXG4iKSkKCgogICAgLyogZG9uZSB3aXRoIFBORyBmaWxlLCBzbyBjbGVhbiB1cCB0byBtaW5pbWl6ZSBtZW1vcnkgdXNhZ2UgKGJ1dCBkbyBOT1QKICAgICAqIG51a2UgaW1hZ2VfZGF0YSEpICovCgogICAgcmVhZHBuZ19jbGVhbnVwKEZBTFNFKTsKICAgIGZjbG9zZShpbmZpbGUpOwoKICAgIGlmICghaW1hZ2VfZGF0YSkgewogICAgICAgIGZwcmludGYoc3RkZXJyLCBQUk9HTkFNRSAiOiAgdW5hYmxlIHRvIGRlY29kZSBQTkcgaW1hZ2VcbiIpOwogICAgICAgIGV4aXQoMyk7CiAgICB9CgoKICAgIC8qIGRpc3BsYXkgaW1hZ2UgKGNvbXBvc2l0ZSB3aXRoIGJhY2tncm91bmQgaWYgcmVxdWVzdGVkKSAqLwoKICAgIFRyYWNlKChzdGRlcnIsICJjYWxsaW5nIHJwbmdfeF9kaXNwbGF5X2ltYWdlKClcbiIpKQogICAgaWYgKHJwbmdfeF9kaXNwbGF5X2ltYWdlKCkpIHsKICAgICAgICBmcmVlKGltYWdlX2RhdGEpOwogICAgICAgIGV4aXQoNCk7CiAgICB9CiAgICBUcmFjZSgoc3RkZXJyLCAiZG9uZSB3aXRoIHJwbmdfeF9kaXNwbGF5X2ltYWdlKClcbiIpKQoKCiAgICAvKiB3YWl0IGZvciB0aGUgdXNlciB0byB0ZWxsIHVzIHdoZW4gdG8gcXVpdCAqLwoKICAgIHByaW50ZigKICAgICAgIkRvbmUuICBQcmVzcyBRLCBFc2Mgb3IgbW91c2UgYnV0dG9uIDEgKHdpdGhpbiBpbWFnZSB3aW5kb3cpIHRvIHF1aXQuXG4iKTsKICAgIGZmbHVzaChzdGRvdXQpOwoKICAgIGRvCiAgICAgICAgWE5leHRFdmVudChkaXNwbGF5LCAmZSk7CiAgICB3aGlsZSAoIShlLnR5cGUgPT0gQnV0dG9uUHJlc3MgJiYgZS54YnV0dG9uLmJ1dHRvbiA9PSBCdXR0b24xKSAmJgogICAgICAgICAgICEoZS50eXBlID09IEtleVByZXNzICYmICAgIC8qICB2LS0tIG9yIDEgZm9yIHNoaWZ0ZWQga2V5cyAqLwogICAgICAgICAgICAgKChrID0gWExvb2t1cEtleXN5bSgmZS54a2V5LCAwKSkgPT0gWEtfcSB8fCBrID09IFhLX0VzY2FwZSkgKSk7CgoKICAgIC8qIE9LLCB3ZSdyZSBkb25lOiAgY2xlYW4gdXAgYWxsIGltYWdlIGFuZCBYIHJlc291cmNlcyBhbmQgZ28gYXdheSAqLwoKICAgIHJwbmdfeF9jbGVhbnVwKCk7CgogICAgcmV0dXJuIDA7Cn0KCgoKCgpzdGF0aWMgaW50IHJwbmdfeF9jcmVhdGVfd2luZG93KHZvaWQpCnsKICAgIHVjaCAqeGRhdGE7CiAgICBpbnQgbmVlZF9jb2xvcm1hcCA9IEZBTFNFOwogICAgaW50IHNjcmVlbiwgcGFkOwogICAgdWxnIGJnX3BpeGVsID0gMEw7CiAgICB1bGcgYXR0cm1hc2s7CiAgICBXaW5kb3cgcm9vdDsKICAgIFhFdmVudCBlOwogICAgWEdDVmFsdWVzIGdjdmFsdWVzOwogICAgWFNldFdpbmRvd0F0dHJpYnV0ZXMgYXR0cjsKICAgIFhUZXh0UHJvcGVydHkgd2luZG93TmFtZSwgKnBXaW5kb3dOYW1lID0gJndpbmRvd05hbWU7CiAgICBYVGV4dFByb3BlcnR5IGljb25OYW1lLCAqcEljb25OYW1lID0gJmljb25OYW1lOwogICAgWFZpc3VhbEluZm8gdmlzdWFsX2luZm87CiAgICBYU2l6ZUhpbnRzICpzaXplX2hpbnRzOwogICAgWFdNSGludHMgKndtX2hpbnRzOwogICAgWENsYXNzSGludCAqY2xhc3NfaGludHM7CgoKICAgIHNjcmVlbiA9IERlZmF1bHRTY3JlZW4oZGlzcGxheSk7CiAgICBkZXB0aCA9IERpc3BsYXlQbGFuZXMoZGlzcGxheSwgc2NyZWVuKTsKICAgIHJvb3QgPSBSb290V2luZG93KGRpc3BsYXksIHNjcmVlbik7CgojaWZkZWYgREVCVUcKICAgIFhTeW5jaHJvbml6ZShkaXNwbGF5LCBUcnVlKTsKI2VuZGlmCgojaWYgMAovKiBHUlI6ICBhZGQgOC1iaXQgc3VwcG9ydCAqLwogICAgaWYgKC8qIGRlcHRoICE9IDggJiYgKi8gZGVwdGggIT0gMTYgJiYgZGVwdGggIT0gMjQgJiYgZGVwdGggIT0gMzIpIHsKICAgICAgICBmcHJpbnRmKHN0ZGVyciwKICAgICAgICAgICJzY3JlZW4gZGVwdGggJWQgbm90IHN1cHBvcnRlZCAob25seSAxNi0sIDI0LSBvciAzMi1iaXQgVHJ1ZUNvbG9yKVxuIiwKICAgICAgICAgIGRlcHRoKTsKICAgICAgICByZXR1cm4gMjsKICAgIH0KCiAgICBYTWF0Y2hWaXN1YWxJbmZvKGRpc3BsYXksIHNjcmVlbiwgZGVwdGgsCiAgICAgIChkZXB0aCA9PSA4KT8gUHNldWRvQ29sb3IgOiBUcnVlQ29sb3IsICZ2aXN1YWxfaW5mbyk7CiAgICB2aXN1YWwgPSB2aXN1YWxfaW5mby52aXN1YWw7CiNlbHNlCiAgICBpZiAoZGVwdGggIT0gMTYgJiYgZGVwdGggIT0gMjQgJiYgZGVwdGggIT0gMzIpIHsKICAgICAgICBpbnQgdmlzdWFsc19tYXRjaGVkID0gMDsKCiAgICAgICAgVHJhY2UoKHN0ZGVyciwgImRlZmF1bHQgZGVwdGggaXMgJWQ6ICBjaGVja2luZyBvdGhlciB2aXN1YWxzXG4iLAogICAgICAgICAgZGVwdGgpKQoKICAgICAgICAvKiAyNC1iaXQgZmlyc3QgKi8KICAgICAgICB2aXN1YWxfaW5mby5zY3JlZW4gPSBzY3JlZW47CiAgICAgICAgdmlzdWFsX2luZm8uZGVwdGggPSAyNDsKICAgICAgICB2aXN1YWxfbGlzdCA9IFhHZXRWaXN1YWxJbmZvKGRpc3BsYXksCiAgICAgICAgICBWaXN1YWxTY3JlZW5NYXNrIHwgVmlzdWFsRGVwdGhNYXNrLCAmdmlzdWFsX2luZm8sICZ2aXN1YWxzX21hdGNoZWQpOwogICAgICAgIGlmICh2aXN1YWxzX21hdGNoZWQgPT0gMCkgewovKiBHUlI6ICBhZGQgMTUtLCAxNi0gYW5kIDMyLWJpdCBUcnVlQ29sb3IgdmlzdWFscyAoYWxzbyBEaXJlY3RDb2xvcj8pICovCiAgICAgICAgICAgIGZwcmludGYoc3RkZXJyLCAiZGVmYXVsdCBzY3JlZW4gZGVwdGggJWQgbm90IHN1cHBvcnRlZCwgYW5kIG5vIgogICAgICAgICAgICAgICIgMjQtYml0IHZpc3VhbHMgZm91bmRcbiIsIGRlcHRoKTsKICAgICAgICAgICAgcmV0dXJuIDI7CiAgICAgICAgfQogICAgICAgIFRyYWNlKChzdGRlcnIsICJYR2V0VmlzdWFsSW5mbygpIHJldHVybmVkICVkIDI0LWJpdCB2aXN1YWxzXG4iLAogICAgICAgICAgdmlzdWFsc19tYXRjaGVkKSkKICAgICAgICB2aXN1YWwgPSB2aXN1YWxfbGlzdFswXS52aXN1YWw7CiAgICAgICAgZGVwdGggPSB2aXN1YWxfbGlzdFswXS5kZXB0aDsKLyoKICAgICAgICBjb2xvcm1hcF9zaXplID0gdmlzdWFsX2xpc3RbMF0uY29sb3JtYXBfc2l6ZTsKICAgICAgICB2aXN1YWxfY2xhc3MgPSB2aXN1YWwtPmNsYXNzOwogICAgICAgIHZpc3VhbElEID0gWFZpc3VhbElERnJvbVZpc3VhbCh2aXN1YWwpOwogKi8KICAgICAgICBoYXZlX25vbmRlZmF1bHRfdmlzdWFsID0gVFJVRTsKICAgICAgICBuZWVkX2NvbG9ybWFwID0gVFJVRTsKICAgIH0gZWxzZSB7CiAgICAgICAgWE1hdGNoVmlzdWFsSW5mbyhkaXNwbGF5LCBzY3JlZW4sIGRlcHRoLCBUcnVlQ29sb3IsICZ2aXN1YWxfaW5mbyk7CiAgICAgICAgdmlzdWFsID0gdmlzdWFsX2luZm8udmlzdWFsOwogICAgfQojZW5kaWYKCiAgICBSTWFzayA9IHZpc3VhbC0+cmVkX21hc2s7CiAgICBHTWFzayA9IHZpc3VhbC0+Z3JlZW5fbWFzazsKICAgIEJNYXNrID0gdmlzdWFsLT5ibHVlX21hc2s7CgovKiBHUlI6ICBhZGQvY2hlY2sgOC1iaXQgc3VwcG9ydCAqLwogICAgaWYgKGRlcHRoID09IDggfHwgbmVlZF9jb2xvcm1hcCkgewogICAgICAgIGNvbG9ybWFwID0gWENyZWF0ZUNvbG9ybWFwKGRpc3BsYXksIHJvb3QsIHZpc3VhbCwgQWxsb2NOb25lKTsKICAgICAgICBpZiAoIWNvbG9ybWFwKSB7CiAgICAgICAgICAgIGZwcmludGYoc3RkZXJyLCAiWENyZWF0ZUNvbG9ybWFwKCkgZmFpbGVkXG4iKTsKICAgICAgICAgICAgcmV0dXJuIDI7CiAgICAgICAgfQogICAgICAgIGhhdmVfY29sb3JtYXAgPSBUUlVFOwogICAgfQogICAgaWYgKGRlcHRoID09IDE1IHx8IGRlcHRoID09IDE2KSB7CiAgICAgICAgUlNoaWZ0ID0gMTUgLSBycG5nX3hfbXNiKFJNYXNrKTsgICAgLyogdGhlc2UgYXJlIHJpZ2h0LXNoaWZ0cyAqLwogICAgICAgIEdTaGlmdCA9IDE1IC0gcnBuZ194X21zYihHTWFzayk7CiAgICAgICAgQlNoaWZ0ID0gMTUgLSBycG5nX3hfbXNiKEJNYXNrKTsKICAgIH0gZWxzZSBpZiAoZGVwdGggPiAxNikgewojZGVmaW5lIE5PXzI0QklUX01BU0tTCiNpZmRlZiBOT18yNEJJVF9NQVNLUwogICAgICAgIFJTaGlmdCA9IHJwbmdfeF9tc2IoUk1hc2spIC0gNzsgICAgIC8qIHRoZXNlIGFyZSBsZWZ0LXNoaWZ0cyAqLwogICAgICAgIEdTaGlmdCA9IHJwbmdfeF9tc2IoR01hc2spIC0gNzsKICAgICAgICBCU2hpZnQgPSBycG5nX3hfbXNiKEJNYXNrKSAtIDc7CiNlbHNlCiAgICAgICAgUlNoaWZ0ID0gNyAtIHJwbmdfeF9tc2IoUk1hc2spOyAgICAgLyogdGhlc2UgYXJlIHJpZ2h0LXNoaWZ0cywgdG9vICovCiAgICAgICAgR1NoaWZ0ID0gNyAtIHJwbmdfeF9tc2IoR01hc2spOwogICAgICAgIEJTaGlmdCA9IDcgLSBycG5nX3hfbXNiKEJNYXNrKTsKI2VuZGlmCiAgICB9CiAgICBpZiAoZGVwdGggPj0gMTUgJiYgKFJTaGlmdCA8IDAgfHwgR1NoaWZ0IDwgMCB8fCBCU2hpZnQgPCAwKSkgewogICAgICAgIGZwcmludGYoc3RkZXJyLCAicnBuZyBpbnRlcm5hbCBsb2dpYyBlcnJvcjogIG5lZ2F0aXZlIFggc2hpZnQocykhXG4iKTsKICAgICAgICByZXR1cm4gMjsKICAgIH0KCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICBGaW5hbGx5LCBjcmVhdGUgdGhlIHdpbmRvdy4KICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKICAgIGF0dHIuYmFja2luZ19zdG9yZSA9IEFsd2F5czsKICAgIGF0dHIuZXZlbnRfbWFzayA9IEV4cG9zdXJlTWFzayB8IEtleVByZXNzTWFzayB8IEJ1dHRvblByZXNzTWFzazsKICAgIGF0dHJtYXNrID0gQ1dCYWNraW5nU3RvcmUgfCBDV0V2ZW50TWFzazsKICAgIGlmIChoYXZlX25vbmRlZmF1bHRfdmlzdWFsKSB7CiAgICAgICAgYXR0ci5jb2xvcm1hcCA9IGNvbG9ybWFwOwogICAgICAgIGF0dHIuYmFja2dyb3VuZF9waXhlbCA9IDA7CiAgICAgICAgYXR0ci5ib3JkZXJfcGl4ZWwgPSAxOwogICAgICAgIGF0dHJtYXNrIHw9IENXQ29sb3JtYXAgfCBDV0JhY2tQaXhlbCB8IENXQm9yZGVyUGl4ZWw7CiAgICB9CgogICAgd2luZG93ID0gWENyZWF0ZVdpbmRvdyhkaXNwbGF5LCByb290LCAwLCAwLCBpbWFnZV93aWR0aCwgaW1hZ2VfaGVpZ2h0LCAwLAogICAgICBkZXB0aCwgSW5wdXRPdXRwdXQsIHZpc3VhbCwgYXR0cm1hc2ssICZhdHRyKTsKCiAgICBpZiAod2luZG93ID09IE5vbmUpIHsKICAgICAgICBmcHJpbnRmKHN0ZGVyciwgIlhDcmVhdGVXaW5kb3coKSBmYWlsZWRcbiIpOwogICAgICAgIHJldHVybiAyOwogICAgfSBlbHNlCiAgICAgICAgaGF2ZV93aW5kb3cgPSBUUlVFOwoKICAgIGlmIChkZXB0aCA9PSA4KQogICAgICAgIFhTZXRXaW5kb3dDb2xvcm1hcChkaXNwbGF5LCB3aW5kb3csIGNvbG9ybWFwKTsKCiAgICBpZiAoIVhTdHJpbmdMaXN0VG9UZXh0UHJvcGVydHkoJndpbmRvd19uYW1lLCAxLCBwV2luZG93TmFtZSkpCiAgICAgICAgcFdpbmRvd05hbWUgPSBOVUxMOwogICAgaWYgKCFYU3RyaW5nTGlzdFRvVGV4dFByb3BlcnR5KCZpY29uX25hbWUsIDEsIHBJY29uTmFtZSkpCiAgICAgICAgcEljb25OYW1lID0gTlVMTDsKCiAgICAvKiBPSyBpZiBhbnkgaGludHMgYWxsb2NhdGlvbiBmYWlsczsgWFNldFdNUHJvcGVydGllcygpIGFsbG93cyBOVUxMcyAqLwoKICAgIGlmICgoc2l6ZV9oaW50cyA9IFhBbGxvY1NpemVIaW50cygpKSAhPSBOVUxMKSB7CiAgICAgICAgLyogd2luZG93IHdpbGwgbm90IGJlIHJlc2l6YWJsZSAqLwogICAgICAgIHNpemVfaGludHMtPmZsYWdzID0gUE1pblNpemUgfCBQTWF4U2l6ZTsKICAgICAgICBzaXplX2hpbnRzLT5taW5fd2lkdGggPSBzaXplX2hpbnRzLT5tYXhfd2lkdGggPSAoaW50KWltYWdlX3dpZHRoOwogICAgICAgIHNpemVfaGludHMtPm1pbl9oZWlnaHQgPSBzaXplX2hpbnRzLT5tYXhfaGVpZ2h0ID0gKGludClpbWFnZV9oZWlnaHQ7CiAgICB9CgogICAgaWYgKCh3bV9oaW50cyA9IFhBbGxvY1dNSGludHMoKSkgIT0gTlVMTCkgewogICAgICAgIHdtX2hpbnRzLT5pbml0aWFsX3N0YXRlID0gTm9ybWFsU3RhdGU7CiAgICAgICAgd21faGludHMtPmlucHV0ID0gVHJ1ZTsKICAgICAvKiB3bV9oaW50cy0+aWNvbl9waXhtYXAgPSBpY29uX3BpeG1hcDsgKi8KICAgICAgICB3bV9oaW50cy0+ZmxhZ3MgPSBTdGF0ZUhpbnQgfCBJbnB1dEhpbnQgIC8qIHwgSWNvblBpeG1hcEhpbnQgKi8gOwogICAgfQoKICAgIGlmICgoY2xhc3NfaGludHMgPSBYQWxsb2NDbGFzc0hpbnQoKSkgIT0gTlVMTCkgewogICAgICAgIGNsYXNzX2hpbnRzLT5yZXNfbmFtZSA9IHJlc19uYW1lOwogICAgICAgIGNsYXNzX2hpbnRzLT5yZXNfY2xhc3MgPSByZXNfY2xhc3M7CiAgICB9CgogICAgWFNldFdNUHJvcGVydGllcyhkaXNwbGF5LCB3aW5kb3csIHBXaW5kb3dOYW1lLCBwSWNvbk5hbWUsIE5VTEwsIDAsCiAgICAgIHNpemVfaGludHMsIHdtX2hpbnRzLCBjbGFzc19oaW50cyk7CgogICAgLyogdmFyaW91cyBwcm9wZXJ0aWVzIGFuZCBoaW50cyBubyBsb25nZXIgbmVlZGVkOyBmcmVlIG1lbW9yeSAqLwogICAgaWYgKHBXaW5kb3dOYW1lKQogICAgICAgWEZyZWUocFdpbmRvd05hbWUtPnZhbHVlKTsKICAgIGlmIChwSWNvbk5hbWUpCiAgICAgICBYRnJlZShwSWNvbk5hbWUtPnZhbHVlKTsKICAgIGlmIChzaXplX2hpbnRzKQogICAgICAgIFhGcmVlKHNpemVfaGludHMpOwogICAgaWYgKHdtX2hpbnRzKQogICAgICAgWEZyZWUod21faGludHMpOwogICAgaWYgKGNsYXNzX2hpbnRzKQogICAgICAgWEZyZWUoY2xhc3NfaGludHMpOwoKICAgIFhNYXBXaW5kb3coZGlzcGxheSwgd2luZG93KTsKCiAgICBnYyA9IFhDcmVhdGVHQyhkaXNwbGF5LCB3aW5kb3csIDAsICZnY3ZhbHVlcyk7CiAgICBoYXZlX2djID0gVFJVRTsKCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICBGaWxsIHdpbmRvdyB3aXRoIHRoZSBzcGVjaWZpZWQgYmFja2dyb3VuZCBjb2xvci4KICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKICAgIGlmIChkZXB0aCA9PSAyNCB8fCBkZXB0aCA9PSAzMikgewogICAgICAgIGJnX3BpeGVsID0gKCh1bGcpYmdfcmVkICAgPDwgUlNoaWZ0KSB8CiAgICAgICAgICAgICAgICAgICAoKHVsZyliZ19ncmVlbiA8PCBHU2hpZnQpIHwKICAgICAgICAgICAgICAgICAgICgodWxnKWJnX2JsdWUgIDw8IEJTaGlmdCk7CiAgICB9IGVsc2UgaWYgKGRlcHRoID09IDE2KSB7CiAgICAgICAgYmdfcGl4ZWwgPSAoKCgodWxnKWJnX3JlZCAgIDw8IDgpID4+IFJTaGlmdCkgJiBSTWFzaykgfAogICAgICAgICAgICAgICAgICAgKCgoKHVsZyliZ19ncmVlbiA8PCA4KSA+PiBHU2hpZnQpICYgR01hc2spIHwKICAgICAgICAgICAgICAgICAgICgoKCh1bGcpYmdfYmx1ZSAgPDwgOCkgPj4gQlNoaWZ0KSAmIEJNYXNrKTsKICAgIH0gZWxzZSAvKiBkZXB0aCA9PSA4ICovIHsKCiAgICAgICAgLyogR1JSOiAgYWRkIDgtYml0IHN1cHBvcnQgKi8KCiAgICB9CgogICAgWFNldEZvcmVncm91bmQoZGlzcGxheSwgZ2MsIGJnX3BpeGVsKTsKICAgIFhGaWxsUmVjdGFuZ2xlKGRpc3BsYXksIHdpbmRvdywgZ2MsIDAsIDAsIGltYWdlX3dpZHRoLCBpbWFnZV9oZWlnaHQpOwoKLyotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KICAgIFdhaXQgZm9yIGZpcnN0IEV4cG9zZSBldmVudCB0byBkbyBhbnkgZHJhd2luZywgdGhlbiBmbHVzaC4KICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKICAgIGRvCiAgICAgICAgWE5leHRFdmVudChkaXNwbGF5LCAmZSk7CiAgICB3aGlsZSAoZS50eXBlICE9IEV4cG9zZSB8fCBlLnhleHBvc2UuY291bnQpOwoKICAgIFhGbHVzaChkaXNwbGF5KTsKCi8qLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiAgICBBbGxvY2F0ZSBtZW1vcnkgZm9yIHRoZSBYLSBhbmQgZGlzcGxheS1zcGVjaWZpYyB2ZXJzaW9uIG9mIHRoZSBpbWFnZS4KICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qLwoKICAgIGlmIChkZXB0aCA9PSAyNCB8fCBkZXB0aCA9PSAzMikgewogICAgICAgIHhkYXRhID0gKHVjaCAqKW1hbGxvYyg0KmltYWdlX3dpZHRoKmltYWdlX2hlaWdodCk7CiAgICAgICAgcGFkID0gMzI7CiAgICB9IGVsc2UgaWYgKGRlcHRoID09IDE2KSB7CiAgICAgICAgeGRhdGEgPSAodWNoICopbWFsbG9jKDIqaW1hZ2Vfd2lkdGgqaW1hZ2VfaGVpZ2h0KTsKICAgICAgICBwYWQgPSAxNjsKICAgIH0gZWxzZSAvKiBkZXB0aCA9PSA4ICovIHsKICAgICAgICB4ZGF0YSA9ICh1Y2ggKiltYWxsb2MoaW1hZ2Vfd2lkdGgqaW1hZ2VfaGVpZ2h0KTsKICAgICAgICBwYWQgPSA4OwogICAgfQoKICAgIGlmICgheGRhdGEpIHsKICAgICAgICBmcHJpbnRmKHN0ZGVyciwgUFJPR05BTUUgIjogIHVuYWJsZSB0byBhbGxvY2F0ZSBpbWFnZSBtZW1vcnlcbiIpOwogICAgICAgIHJldHVybiA0OwogICAgfQoKICAgIHhpbWFnZSA9IFhDcmVhdGVJbWFnZShkaXNwbGF5LCB2aXN1YWwsIGRlcHRoLCBaUGl4bWFwLCAwLAogICAgICAoY2hhciAqKXhkYXRhLCBpbWFnZV93aWR0aCwgaW1hZ2VfaGVpZ2h0LCBwYWQsIDApOwoKICAgIGlmICgheGltYWdlKSB7CiAgICAgICAgZnByaW50ZihzdGRlcnIsIFBST0dOQU1FICI6ICBYQ3JlYXRlSW1hZ2UoKSBmYWlsZWRcbiIpOwogICAgICAgIGZyZWUoeGRhdGEpOwogICAgICAgIHJldHVybiAzOwogICAgfQoKICAgIC8qIHRvIGF2b2lkIHRlc3RpbmcgdGhlIGJ5dGUgb3JkZXIgZXZlcnkgcGl4ZWwgKG9yIGRvdWJsaW5nIHRoZSBzaXplIG9mCiAgICAgKiB0aGUgZHJhd2luZyByb3V0aW5lIHdpdGggYSBnaWFudCBpZi10ZXN0KSwgd2UgYXJiaXRyYXJpbHkgc2V0IHRoZSBieXRlCiAgICAgKiBvcmRlciB0byBNU0JGaXJzdCBhbmQgbGV0IFhsaWIgd29ycnkgYWJvdXQgaW52ZXJ0aW5nIHRoaW5ncyBvbiBsaXR0bGUtCiAgICAgKiBlbmRpYW4gbWFjaGluZXMgKGxpa2UgTGludXgveDg2LCBvbGQgVkFYZW4sIGV0Yy4pLS10aGlzIGlzIG5vdCB0aGUgbW9zdAogICAgICogZWZmaWNpZW50IGFwcHJvYWNoICh0aGUgZ2lhbnQgaWYtdGVzdCB3b3VsZCBiZSBiZXR0ZXIpLCBidXQgaW4gdGhlCiAgICAgKiBpbnRlcmVzdCBvZiBjbGFyaXR5LCB3ZSB0YWtlIHRoZSBlYXN5IHdheSBvdXQuLi4gKi8KCiAgICB4aW1hZ2UtPmJ5dGVfb3JkZXIgPSBNU0JGaXJzdDsKCiAgICByZXR1cm4gMDsKCn0gLyogZW5kIGZ1bmN0aW9uIHJwbmdfeF9jcmVhdGVfd2luZG93KCkgKi8KCgoKCgpzdGF0aWMgaW50IHJwbmdfeF9kaXNwbGF5X2ltYWdlKHZvaWQpCnsKICAgIHVjaCAqc3JjOwogICAgY2hhciAqZGVzdDsKICAgIHVjaCByLCBnLCBiLCBhOwogICAgdWxnIGksIHJvdywgbGFzdHJvdyA9IDA7CiAgICB1bGcgcGl4ZWw7CiAgICBpbnQgeGltYWdlX3Jvd2J5dGVzID0geGltYWdlLT5ieXRlc19wZXJfbGluZTsKLyogIGludCBicHAgPSB4aW1hZ2UtPmJpdHNfcGVyX3BpeGVsOyAgKi8KCgogICAgVHJhY2UoKHN0ZGVyciwgImJlZ2lubmluZyBkaXNwbGF5IGxvb3AgKGltYWdlX2NoYW5uZWxzID09ICVkKVxuIiwKICAgICAgaW1hZ2VfY2hhbm5lbHMpKQogICAgVHJhY2UoKHN0ZGVyciwgIiAgICh3aWR0aCA9ICVsZCwgcm93Ynl0ZXMgPSAlbGQsIHhpbWFnZV9yb3dieXRlcyA9ICVkKVxuIiwKICAgICAgaW1hZ2Vfd2lkdGgsIGltYWdlX3Jvd2J5dGVzLCB4aW1hZ2Vfcm93Ynl0ZXMpKQogICAgVHJhY2UoKHN0ZGVyciwgIiAgIChicHAgPSAlZClcbiIsIHhpbWFnZS0+Yml0c19wZXJfcGl4ZWwpKQogICAgVHJhY2UoKHN0ZGVyciwgIiAgIChieXRlX29yZGVyID0gJXMpXG4iLCB4aW1hZ2UtPmJ5dGVfb3JkZXIgPT0gTVNCRmlyc3Q/CiAgICAgICJNU0JGaXJzdCIgOiAoeGltYWdlLT5ieXRlX29yZGVyID09IExTQkZpcnN0PyAiTFNCRmlyc3QiIDogInVua25vd24iKSkpCgogICAgaWYgKGRlcHRoID09IDI0IHx8IGRlcHRoID09IDMyKSB7CiAgICAgICAgdWxnIHJlZCwgZ3JlZW4sIGJsdWU7CgogICAgICAgIGZvciAobGFzdHJvdyA9IHJvdyA9IDA7ICByb3cgPCBpbWFnZV9oZWlnaHQ7ICArK3JvdykgewogICAgICAgICAgICBzcmMgPSBpbWFnZV9kYXRhICsgcm93KmltYWdlX3Jvd2J5dGVzOwogICAgICAgICAgICBkZXN0ID0geGltYWdlLT5kYXRhICsgcm93KnhpbWFnZV9yb3dieXRlczsKICAgICAgICAgICAgaWYgKGltYWdlX2NoYW5uZWxzID09IDMpIHsKICAgICAgICAgICAgICAgIGZvciAoaSA9IGltYWdlX3dpZHRoOyAgaSA+IDA7ICAtLWkpIHsKICAgICAgICAgICAgICAgICAgICByZWQgICA9ICpzcmMrKzsKICAgICAgICAgICAgICAgICAgICBncmVlbiA9ICpzcmMrKzsKICAgICAgICAgICAgICAgICAgICBibHVlICA9ICpzcmMrKzsKI2lmZGVmIE5PXzI0QklUX01BU0tTCiAgICAgICAgICAgICAgICAgICAgcGl4ZWwgPSAocmVkICAgPDwgUlNoaWZ0KSB8CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZ3JlZW4gPDwgR1NoaWZ0KSB8CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoYmx1ZSAgPDwgQlNoaWZ0KTsKICAgICAgICAgICAgICAgICAgICAvKiByZWNhbGwgdGhhdCB3ZSBzZXQgeGltYWdlLT5ieXRlX29yZGVyID0gTVNCRmlyc3QgYWJvdmUgKi8KICAgICAgICAgICAgICAgICAgICAvKiBHUlIgQlVHOiAgdGhpcyBhc3N1bWVzIGJwcCA9PSAzMiwgYnV0IG1heSBiZSAyNDogKi8KICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAyNCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAxNikgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAgOCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKCBwaXhlbCAgICAgICAgJiAweGZmKTsKI2Vsc2UKICAgICAgICAgICAgICAgICAgICByZWQgICA9IChSU2hpZnQgPCAwKT8gcmVkICAgPDwgKC1SU2hpZnQpIDogcmVkICAgPj4gUlNoaWZ0OwogICAgICAgICAgICAgICAgICAgIGdyZWVuID0gKEdTaGlmdCA8IDApPyBncmVlbiA8PCAoLUdTaGlmdCkgOiBncmVlbiA+PiBHU2hpZnQ7CiAgICAgICAgICAgICAgICAgICAgYmx1ZSAgPSAoQlNoaWZ0IDwgMCk/IGJsdWUgIDw8ICgtQlNoaWZ0KSA6IGJsdWUgID4+IEJTaGlmdDsKICAgICAgICAgICAgICAgICAgICBwaXhlbCA9IChyZWQgJiBSTWFzaykgfCAoZ3JlZW4gJiBHTWFzaykgfCAoYmx1ZSAmIEJNYXNrKTsKICAgICAgICAgICAgICAgICAgICAvKiByZWNhbGwgdGhhdCB3ZSBzZXQgeGltYWdlLT5ieXRlX29yZGVyID0gTVNCRmlyc3QgYWJvdmUgKi8KICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAyNCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAxNikgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAgOCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKCBwaXhlbCAgICAgICAgJiAweGZmKTsKI2VuZGlmCiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0gZWxzZSAvKiBpZiAoaW1hZ2VfY2hhbm5lbHMgPT0gNCkgKi8gewogICAgICAgICAgICAgICAgZm9yIChpID0gaW1hZ2Vfd2lkdGg7ICBpID4gMDsgIC0taSkgewogICAgICAgICAgICAgICAgICAgIHIgPSAqc3JjKys7CiAgICAgICAgICAgICAgICAgICAgZyA9ICpzcmMrKzsKICAgICAgICAgICAgICAgICAgICBiID0gKnNyYysrOwogICAgICAgICAgICAgICAgICAgIGEgPSAqc3JjKys7CiAgICAgICAgICAgICAgICAgICAgaWYgKGEgPT0gMjU1KSB7CiAgICAgICAgICAgICAgICAgICAgICAgIHJlZCAgID0gcjsKICAgICAgICAgICAgICAgICAgICAgICAgZ3JlZW4gPSBnOwogICAgICAgICAgICAgICAgICAgICAgICBibHVlICA9IGI7CiAgICAgICAgICAgICAgICAgICAgfSBlbHNlIGlmIChhID09IDApIHsKICAgICAgICAgICAgICAgICAgICAgICAgcmVkICAgPSBiZ19yZWQ7CiAgICAgICAgICAgICAgICAgICAgICAgIGdyZWVuID0gYmdfZ3JlZW47CiAgICAgICAgICAgICAgICAgICAgICAgIGJsdWUgID0gYmdfYmx1ZTsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgewogICAgICAgICAgICAgICAgICAgICAgICAvKiB0aGlzIG1hY3JvIChmcm9tIHBuZy5oKSBjb21wb3NpdGVzIHRoZSBmb3JlZ3JvdW5kCiAgICAgICAgICAgICAgICAgICAgICAgICAqIGFuZCBiYWNrZ3JvdW5kIHZhbHVlcyBhbmQgcHV0cyB0aGUgcmVzdWx0IGludG8gdGhlCiAgICAgICAgICAgICAgICAgICAgICAgICAqIGZpcnN0IGFyZ3VtZW50ICovCiAgICAgICAgICAgICAgICAgICAgICAgIGFscGhhX2NvbXBvc2l0ZShyZWQsICAgciwgYSwgYmdfcmVkKTsKICAgICAgICAgICAgICAgICAgICAgICAgYWxwaGFfY29tcG9zaXRlKGdyZWVuLCBnLCBhLCBiZ19ncmVlbik7CiAgICAgICAgICAgICAgICAgICAgICAgIGFscGhhX2NvbXBvc2l0ZShibHVlLCAgYiwgYSwgYmdfYmx1ZSk7CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgIHBpeGVsID0gKHJlZCAgIDw8IFJTaGlmdCkgfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKGdyZWVuIDw8IEdTaGlmdCkgfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKGJsdWUgIDw8IEJTaGlmdCk7CiAgICAgICAgICAgICAgICAgICAgLyogcmVjYWxsIHRoYXQgd2Ugc2V0IHhpbWFnZS0+Ynl0ZV9vcmRlciA9IE1TQkZpcnN0IGFib3ZlICovCiAgICAgICAgICAgICAgICAgICAgKmRlc3QrKyA9IChjaGFyKSgocGl4ZWwgPj4gMjQpICYgMHhmZik7CiAgICAgICAgICAgICAgICAgICAgKmRlc3QrKyA9IChjaGFyKSgocGl4ZWwgPj4gMTYpICYgMHhmZik7CiAgICAgICAgICAgICAgICAgICAgKmRlc3QrKyA9IChjaGFyKSgocGl4ZWwgPj4gIDgpICYgMHhmZik7CiAgICAgICAgICAgICAgICAgICAgKmRlc3QrKyA9IChjaGFyKSggcGl4ZWwgICAgICAgICYgMHhmZik7CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgICAgLyogZGlzcGxheSBhZnRlciBldmVyeSAxNiBsaW5lcyAqLwogICAgICAgICAgICBpZiAoKChyb3crMSkgJiAweGYpID09IDApIHsKICAgICAgICAgICAgICAgIFhQdXRJbWFnZShkaXNwbGF5LCB3aW5kb3csIGdjLCB4aW1hZ2UsIDAsIChpbnQpbGFzdHJvdywgMCwKICAgICAgICAgICAgICAgICAgKGludClsYXN0cm93LCBpbWFnZV93aWR0aCwgMTYpOwogICAgICAgICAgICAgICAgWEZsdXNoKGRpc3BsYXkpOwogICAgICAgICAgICAgICAgbGFzdHJvdyA9IHJvdyArIDE7CiAgICAgICAgICAgIH0KICAgICAgICB9CgogICAgfSBlbHNlIGlmIChkZXB0aCA9PSAxNikgewogICAgICAgIHVzaCByZWQsIGdyZWVuLCBibHVlOwoKICAgICAgICBmb3IgKGxhc3Ryb3cgPSByb3cgPSAwOyAgcm93IDwgaW1hZ2VfaGVpZ2h0OyAgKytyb3cpIHsKICAgICAgICAgICAgc3JjID0gaW1hZ2VfZGF0YSArIHJvdyppbWFnZV9yb3dieXRlczsKICAgICAgICAgICAgZGVzdCA9IHhpbWFnZS0+ZGF0YSArIHJvdyp4aW1hZ2Vfcm93Ynl0ZXM7CiAgICAgICAgICAgIGlmIChpbWFnZV9jaGFubmVscyA9PSAzKSB7CiAgICAgICAgICAgICAgICBmb3IgKGkgPSBpbWFnZV93aWR0aDsgIGkgPiAwOyAgLS1pKSB7CiAgICAgICAgICAgICAgICAgICAgcmVkICAgPSAoKHVzaCkoKnNyYykgPDwgOCk7CiAgICAgICAgICAgICAgICAgICAgKytzcmM7CiAgICAgICAgICAgICAgICAgICAgZ3JlZW4gPSAoKHVzaCkoKnNyYykgPDwgOCk7CiAgICAgICAgICAgICAgICAgICAgKytzcmM7CiAgICAgICAgICAgICAgICAgICAgYmx1ZSAgPSAoKHVzaCkoKnNyYykgPDwgOCk7CiAgICAgICAgICAgICAgICAgICAgKytzcmM7CiAgICAgICAgICAgICAgICAgICAgcGl4ZWwgPSAoKHJlZCAgID4+IFJTaGlmdCkgJiBSTWFzaykgfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKChncmVlbiA+PiBHU2hpZnQpICYgR01hc2spIHwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICgoYmx1ZSAgPj4gQlNoaWZ0KSAmIEJNYXNrKTsKICAgICAgICAgICAgICAgICAgICAvKiByZWNhbGwgdGhhdCB3ZSBzZXQgeGltYWdlLT5ieXRlX29yZGVyID0gTVNCRmlyc3QgYWJvdmUgKi8KICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAgOCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKCBwaXhlbCAgICAgICAgJiAweGZmKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfSBlbHNlIC8qIGlmIChpbWFnZV9jaGFubmVscyA9PSA0KSAqLyB7CiAgICAgICAgICAgICAgICBmb3IgKGkgPSBpbWFnZV93aWR0aDsgIGkgPiAwOyAgLS1pKSB7CiAgICAgICAgICAgICAgICAgICAgciA9ICpzcmMrKzsKICAgICAgICAgICAgICAgICAgICBnID0gKnNyYysrOwogICAgICAgICAgICAgICAgICAgIGIgPSAqc3JjKys7CiAgICAgICAgICAgICAgICAgICAgYSA9ICpzcmMrKzsKICAgICAgICAgICAgICAgICAgICBpZiAoYSA9PSAyNTUpIHsKICAgICAgICAgICAgICAgICAgICAgICAgcmVkICAgPSAoKHVzaClyIDw8IDgpOwogICAgICAgICAgICAgICAgICAgICAgICBncmVlbiA9ICgodXNoKWcgPDwgOCk7CiAgICAgICAgICAgICAgICAgICAgICAgIGJsdWUgID0gKCh1c2gpYiA8PCA4KTsKICAgICAgICAgICAgICAgICAgICB9IGVsc2UgaWYgKGEgPT0gMCkgewogICAgICAgICAgICAgICAgICAgICAgICByZWQgICA9ICgodXNoKWJnX3JlZCAgIDw8IDgpOwogICAgICAgICAgICAgICAgICAgICAgICBncmVlbiA9ICgodXNoKWJnX2dyZWVuIDw8IDgpOwogICAgICAgICAgICAgICAgICAgICAgICBibHVlICA9ICgodXNoKWJnX2JsdWUgIDw8IDgpOwogICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICAgICAgICAgIC8qIHRoaXMgbWFjcm8gKGZyb20gcG5nLmgpIGNvbXBvc2l0ZXMgdGhlIGZvcmVncm91bmQKICAgICAgICAgICAgICAgICAgICAgICAgICogYW5kIGJhY2tncm91bmQgdmFsdWVzIGFuZCBwdXRzIHRoZSByZXN1bHQgYmFjayBpbnRvCiAgICAgICAgICAgICAgICAgICAgICAgICAqIHRoZSBmaXJzdCBhcmd1bWVudCAoPT0gZmcgYnl0ZSBoZXJlOiAgc2FmZSkgKi8KICAgICAgICAgICAgICAgICAgICAgICAgYWxwaGFfY29tcG9zaXRlKHIsIHIsIGEsIGJnX3JlZCk7CiAgICAgICAgICAgICAgICAgICAgICAgIGFscGhhX2NvbXBvc2l0ZShnLCBnLCBhLCBiZ19ncmVlbik7CiAgICAgICAgICAgICAgICAgICAgICAgIGFscGhhX2NvbXBvc2l0ZShiLCBiLCBhLCBiZ19ibHVlKTsKICAgICAgICAgICAgICAgICAgICAgICAgcmVkICAgPSAoKHVzaClyIDw8IDgpOwogICAgICAgICAgICAgICAgICAgICAgICBncmVlbiA9ICgodXNoKWcgPDwgOCk7CiAgICAgICAgICAgICAgICAgICAgICAgIGJsdWUgID0gKCh1c2gpYiA8PCA4KTsKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgcGl4ZWwgPSAoKHJlZCAgID4+IFJTaGlmdCkgJiBSTWFzaykgfAogICAgICAgICAgICAgICAgICAgICAgICAgICAgKChncmVlbiA+PiBHU2hpZnQpICYgR01hc2spIHwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICgoYmx1ZSAgPj4gQlNoaWZ0KSAmIEJNYXNrKTsKICAgICAgICAgICAgICAgICAgICAvKiByZWNhbGwgdGhhdCB3ZSBzZXQgeGltYWdlLT5ieXRlX29yZGVyID0gTVNCRmlyc3QgYWJvdmUgKi8KICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKChwaXhlbCA+PiAgOCkgJiAweGZmKTsKICAgICAgICAgICAgICAgICAgICAqZGVzdCsrID0gKGNoYXIpKCBwaXhlbCAgICAgICAgJiAweGZmKTsKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgICAgICAvKiBkaXNwbGF5IGFmdGVyIGV2ZXJ5IDE2IGxpbmVzICovCiAgICAgICAgICAgIGlmICgoKHJvdysxKSAmIDB4ZikgPT0gMCkgewogICAgICAgICAgICAgICAgWFB1dEltYWdlKGRpc3BsYXksIHdpbmRvdywgZ2MsIHhpbWFnZSwgMCwgKGludClsYXN0cm93LCAwLAogICAgICAgICAgICAgICAgICAoaW50KWxhc3Ryb3csIGltYWdlX3dpZHRoLCAxNik7CiAgICAgICAgICAgICAgICBYRmx1c2goZGlzcGxheSk7CiAgICAgICAgICAgICAgICBsYXN0cm93ID0gcm93ICsgMTsKICAgICAgICAgICAgfQogICAgICAgIH0KCiAgICB9IGVsc2UgLyogZGVwdGggPT0gOCAqLyB7CgogICAgICAgIC8qIEdSUjogIGFkZCA4LWJpdCBzdXBwb3J0ICovCgogICAgfQoKICAgIFRyYWNlKChzdGRlcnIsICJjYWxsaW5nIGZpbmFsIFhQdXRJbWFnZSgpXG4iKSkKICAgIGlmIChsYXN0cm93IDwgaW1hZ2VfaGVpZ2h0KSB7CiAgICAgICAgWFB1dEltYWdlKGRpc3BsYXksIHdpbmRvdywgZ2MsIHhpbWFnZSwgMCwgKGludClsYXN0cm93LCAwLAogICAgICAgICAgKGludClsYXN0cm93LCBpbWFnZV93aWR0aCwgaW1hZ2VfaGVpZ2h0LWxhc3Ryb3cpOwogICAgICAgIFhGbHVzaChkaXNwbGF5KTsKICAgIH0KCiAgICByZXR1cm4gMDsKfQoKCgoKc3RhdGljIHZvaWQgcnBuZ194X2NsZWFudXAodm9pZCkKewogICAgaWYgKGltYWdlX2RhdGEpIHsKICAgICAgICBmcmVlKGltYWdlX2RhdGEpOwogICAgICAgIGltYWdlX2RhdGEgPSBOVUxMOwogICAgfQoKICAgIGlmICh4aW1hZ2UpIHsKICAgICAgICBpZiAoeGltYWdlLT5kYXRhKSB7CiAgICAgICAgICAgIGZyZWUoeGltYWdlLT5kYXRhKTsgICAgICAgICAgIC8qIHdlIGFsbG9jYXRlZCBpdCwgc28gd2UgZnJlZSBpdCAqLwogICAgICAgICAgICB4aW1hZ2UtPmRhdGEgPSAoY2hhciAqKU5VTEw7ICAvKiAgaW5zdGVhZCBvZiBYRGVzdHJveUltYWdlKCkgKi8KICAgICAgICB9CiAgICAgICAgWERlc3Ryb3lJbWFnZSh4aW1hZ2UpOwogICAgICAgIHhpbWFnZSA9IE5VTEw7CiAgICB9CgogICAgaWYgKGhhdmVfZ2MpCiAgICAgICAgWEZyZWVHQyhkaXNwbGF5LCBnYyk7CgogICAgaWYgKGhhdmVfd2luZG93KQogICAgICAgIFhEZXN0cm95V2luZG93KGRpc3BsYXksIHdpbmRvdyk7CgogICAgaWYgKGhhdmVfY29sb3JtYXApCiAgICAgICAgWEZyZWVDb2xvcm1hcChkaXNwbGF5LCBjb2xvcm1hcCk7CgogICAgaWYgKGhhdmVfbm9uZGVmYXVsdF92aXN1YWwpCiAgICAgICAgWEZyZWUodmlzdWFsX2xpc3QpOwp9CgoKCgoKc3RhdGljIGludCBycG5nX3hfbXNiKHVsZyB1MzJ2YWwpCnsKICAgIGludCBpOwoKICAgIGZvciAoaSA9IDMxOyAgaSA+PSAwOyAgLS1pKSB7CiAgICAgICAgaWYgKHUzMnZhbCAmIDB4ODAwMDAwMDBMKQogICAgICAgICAgICBicmVhazsKICAgICAgICB1MzJ2YWwgPDw9IDE7CiAgICB9CiAgICByZXR1cm4gaTsKfQo=