LyogLSotIHRhYi13aWR0aDogODsgYy1iYXNpYy1vZmZzZXQ6IDQgLSotICovCi8qCiAqIEFuaW1hdGlvbiBjb250cm9sCiAqCiAqIENvcHlyaWdodCAxOTk4LCAxOTk5IEVyaWMgS29obAogKiAJCSAgIDE5OTkgRXJpYyBQb3VlY2gKICoKICogTk9URVMKICogICBJIHdpbGwgb25seSBpbXByb3ZlIHRoaXMgY29udHJvbCBvbmNlIGluIGEgd2hpbGUuCiAqICAgICBFcmljIDxla29obEBhYm8ucmhlaW4temVpdHVuZy5kZT4KICoKICogVE9ETzoKICogICAtIGNoZWNrIGZvciB0aGUgJ3JlYyAnIGxpc3QgaW4gc29tZSBBVkkgZmlsZXMKICogICAtIGltcGxlbWVudCBzb21lIG1pc3NpbmcgZmxhZ3MgKEFDU19UUkFOU1BBUkVOVCBhbmQgQUNTX0NFTlRFUikKICogICAtIHByb3RlY3Rpb24gYmV0d2VlbiBzZXJ2aWNlIHRocmVhZCBhbmQgd25kcHJvYyBtZXNzYWdlcyBoYW5kbGluZyAKICogICAgIGNvbmN1cnJlbnQgYWNjZXNzIHRvIGluZm9QdHIKICovCgoKI2luY2x1ZGUgIndpbmJhc2UuaCIKI2luY2x1ZGUgImNvbW1jdHJsLmgiCiNpbmNsdWRlICJkcml2ZXIuaCIKI2luY2x1ZGUgImFuaW1hdGUuaCIKI2luY2x1ZGUgIm1tc3lzdGVtLmgiCiNpbmNsdWRlICJzZXJ2aWNlcy5oIgojaW5jbHVkZSAiZGVidWd0b29scy5oIgoKREVGQVVMVF9ERUJVR19DSEFOTkVMKGFuaW1hdGUpCgojZGVmaW5lIEFOSU1BVEVfR2V0SW5mb1B0cihoV25kKSAoKEFOSU1BVEVfSU5GTyAqKUdldFdpbmRvd0xvbmdBKGhXbmQsIDApKQoKc3RhdGljIHZvaWQgQU5JTUFURV9Ob3RpZnkoQU5JTUFURV9JTkZPKiBpbmZvUHRyLCBVSU5UIG5vdGlmKQp7CiAgICBTZW5kTWVzc2FnZUEoR2V0UGFyZW50KGluZm9QdHItPmhXbmQpLCBXTV9DT01NQU5ELCAKCQkgTUFLRVdQQVJBTShHZXREbGdDdHJsSUQoaW5mb1B0ci0+aFduZCksIG5vdGlmKSwgCgkJIChMUEFSQU0paW5mb1B0ci0+aFduZCk7Cn0KCnN0YXRpYyBCT09MIEFOSU1BVEVfTG9hZFJlc0EoQU5JTUFURV9JTkZPICppbmZvUHRyLCBISU5TVEFOQ0UgaEluc3QsIExQU1RSIGxwTmFtZSkKewogICAgSFJTUkMgCWhyc3JjOwogICAgTU1JT0lORk8JbW1pbmZvOwogICAgTFBWT0lECWxwQXZpOwogICAgCiAgICBocnNyYyA9IEZpbmRSZXNvdXJjZUEoaEluc3QsIGxwTmFtZSwgIkFWSSIpOwogICAgaWYgKCFocnNyYykKCXJldHVybiBGQUxTRTsKICAgIAogICAgaW5mb1B0ci0+aFJlcyA9IExvYWRSZXNvdXJjZShoSW5zdCwgaHJzcmMpOwogICAgaWYgKCFpbmZvUHRyLT5oUmVzKQogCXJldHVybiBGQUxTRTsKICAgIAogICAgbHBBdmkgPSBMb2NrUmVzb3VyY2UoaW5mb1B0ci0+aFJlcyk7CiAgICBpZiAoIWxwQXZpKQoJcmV0dXJuIEZBTFNFOwogICAgCiAgICBtZW1zZXQoJm1taW5mbywgMCwgc2l6ZW9mKG1taW5mbykpOwogICAgbW1pbmZvLmZjY0lPUHJvYyA9IEZPVVJDQ19NRU07CiAgICBtbWluZm8ucGNoQnVmZmVyID0gKExQU1RSKWxwQXZpOwogICAgbW1pbmZvLmNjaEJ1ZmZlciA9IFNpemVvZlJlc291cmNlKGhJbnN0LCBocnNyYyk7CiAgICBpbmZvUHRyLT5oTU1pbyA9IG1taW9PcGVuQShOVUxMLCAmbW1pbmZvLCBNTUlPX1JFQUQpOwogICAgCiAgICBpZiAoIWluZm9QdHItPmhNTWlvKSB7CglHbG9iYWxGcmVlKChIR0xPQkFMKWxwQXZpKTsKCXJldHVybiBGQUxTRTsKICAgIH0KICAgIAogICAgcmV0dXJuIFRSVUU7Cn0KCgpzdGF0aWMgQk9PTCBBTklNQVRFX0xvYWRGaWxlQShBTklNQVRFX0lORk8gKmluZm9QdHIsIExQU1RSIGxwTmFtZSkKewogICAgaW5mb1B0ci0+aE1NaW8gPSBtbWlvT3BlbkEoKExQU1RSKWxwTmFtZSwgTlVMTCwKCQkJICAgICAgIE1NSU9fQUxMT0NCVUYgfCBNTUlPX1JFQUQgfCBNTUlPX0RFTllXUklURSk7CiAgICAKICAgIGlmICghaW5mb1B0ci0+aE1NaW8pCglyZXR1cm4gRkFMU0U7CiAgICAKICAgIHJldHVybiBUUlVFOwp9CgoKc3RhdGljIExSRVNVTFQgQU5JTUFURV9Eb1N0b3AoQU5JTUFURV9JTkZPICppbmZvUHRyKQp7CiAgICBFbnRlckNyaXRpY2FsU2VjdGlvbigmaW5mb1B0ci0+Y3MpOwoKICAgIC8qIHNob3VsZCBzdG9wIHBsYXlpbmcgKi8KICAgIGlmIChpbmZvUHRyLT5oU2VydmljZSkgewoJU0VSVklDRV9EZWxldGUoaW5mb1B0ci0+aFNlcnZpY2UpOwoJaW5mb1B0ci0+aFNlcnZpY2UgPSAwOwogICAgfQogICAgaWYgKGluZm9QdHItPnVUaW1lcikgewoJS2lsbFRpbWVyKGluZm9QdHItPmhXbmQsIGluZm9QdHItPnVUaW1lcik7CglpbmZvUHRyLT51VGltZXIgPSAwOwogICAgfQoKICAgIExlYXZlQ3JpdGljYWxTZWN0aW9uKCZpbmZvUHRyLT5jcyk7CgogICAgQU5JTUFURV9Ob3RpZnkoaW5mb1B0ciwgQUNOX1NUT1ApOwoKICAgIHJldHVybiBUUlVFOwp9CgoKc3RhdGljIHZvaWQgQU5JTUFURV9GcmVlKEFOSU1BVEVfSU5GTyAqaW5mb1B0cikKewogICAgaWYgKGluZm9QdHItPmhNTWlvKSB7CglBTklNQVRFX0RvU3RvcChpbmZvUHRyKTsKCW1taW9DbG9zZShpbmZvUHRyLT5oTU1pbywgMCk7CglpZiAoaW5mb1B0ci0+aFJlcykgewogCSAgICBGcmVlUmVzb3VyY2UoaW5mb1B0ci0+aFJlcyk7CgkgICAgaW5mb1B0ci0+aFJlcyA9IDA7Cgl9CglpZiAoaW5mb1B0ci0+bHBJbmRleCkgewoJICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGluZm9QdHItPmxwSW5kZXgpOwoJICAgIGluZm9QdHItPmxwSW5kZXggPSBOVUxMOwoJfQoJaWYgKGluZm9QdHItPmhpYykgewoJICAgIChpbmZvUHRyLT5mbklDQ2xvc2UpKGluZm9QdHItPmhpYyk7CgkgICAgaW5mb1B0ci0+aGljID0gMDsKCX0KCWlmIChpbmZvUHRyLT5pbmJpaCkgewoJICAgIEhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGluZm9QdHItPmluYmloKTsKCSAgICBpbmZvUHRyLT5pbmJpaCA9IE5VTEw7Cgl9CglpZiAoaW5mb1B0ci0+b3V0YmloKSB7CgkgICAgSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgaW5mb1B0ci0+b3V0YmloKTsKCSAgICBpbmZvUHRyLT5vdXRiaWggPSBOVUxMOwoJfQoJSGVhcEZyZWUoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgaW5mb1B0ci0+aW5kYXRhKTsKCUhlYXBGcmVlKEdldFByb2Nlc3NIZWFwKCksIDAsIGluZm9QdHItPm91dGRhdGEpOwoJaW5mb1B0ci0+aW5kYXRhID0gaW5mb1B0ci0+b3V0ZGF0YSA9IE5VTEw7CglpbmZvUHRyLT5oV25kID0gMDsKCWluZm9QdHItPmhNTWlvID0gMDsKCW1lbXNldCgmaW5mb1B0ci0+bWFoLCAwLCBzaXplb2YoaW5mb1B0ci0+bWFoKSk7CgltZW1zZXQoJmluZm9QdHItPmFzaCwgMCwgc2l6ZW9mKGluZm9QdHItPmFzaCkpOwoJaW5mb1B0ci0+bkZyb21GcmFtZSA9IGluZm9QdHItPm5Ub0ZyYW1lID0gaW5mb1B0ci0+bkxvb3AgPSBpbmZvUHRyLT5jdXJyRnJhbWUgPSAwOwogICAgfQp9CgoKc3RhdGljIExSRVNVTFQgQU5JTUFURV9QYWludEZyYW1lKEFOSU1BVEVfSU5GTyogaW5mb1B0ciwgSERDIGhEQykKewogICAgaWYgKCFoREMgfHwgIWluZm9QdHItPmluYmloKQoJcmV0dXJuIFRSVUU7CiAgICBpZiAoaW5mb1B0ci0+aGljKQoJU3RyZXRjaERJQml0cyhoREMsIDAsIDAsIGluZm9QdHItPm91dGJpaC0+YmlXaWR0aCwgaW5mb1B0ci0+b3V0YmloLT5iaUhlaWdodCwgCgkJICAgICAgMCwgMCwgaW5mb1B0ci0+b3V0YmloLT5iaVdpZHRoLCBpbmZvUHRyLT5vdXRiaWgtPmJpSGVpZ2h0LCAKCQkgICAgICBpbmZvUHRyLT5vdXRkYXRhLCAoTFBCSVRNQVBJTkZPKWluZm9QdHItPm91dGJpaCwgRElCX1JHQl9DT0xPUlMsIAoJCSAgICAgIFNSQ0NPUFkpOwogICAgZWxzZQoJU3RyZXRjaERJQml0cyhoREMsIDAsIDAsIGluZm9QdHItPmluYmloLT5iaVdpZHRoLCBpbmZvUHRyLT5pbmJpaC0+YmlIZWlnaHQsIAoJCSAgICAgIDAsIDAsIGluZm9QdHItPmluYmloLT5iaVdpZHRoLCBpbmZvUHRyLT5pbmJpaC0+YmlIZWlnaHQsIAoJCSAgICAgIGluZm9QdHItPmluZGF0YSwgKExQQklUTUFQSU5GTylpbmZvUHRyLT5pbmJpaCwgRElCX1JHQl9DT0xPUlMsIAoJCSAgICAgIFNSQ0NPUFkpOwoKICAgIHJldHVybiBUUlVFOwp9CgpzdGF0aWMgTFJFU1VMVCBBTklNQVRFX0RyYXdGcmFtZShBTklNQVRFX0lORk8qIGluZm9QdHIpCnsKICAgIEhEQwkJaERDOwoKICAgIFRSQUNFKCJEcmF3aW5nIGZyYW1lICVkIChsb29wICVkKVxuIiwgaW5mb1B0ci0+Y3VyckZyYW1lLCBpbmZvUHRyLT5uTG9vcCk7CgogICAgRW50ZXJDcml0aWNhbFNlY3Rpb24oJmluZm9QdHItPmNzKTsKCiAgICBtbWlvU2VlayhpbmZvUHRyLT5oTU1pbywgaW5mb1B0ci0+bHBJbmRleFtpbmZvUHRyLT5jdXJyRnJhbWVdLCBTRUVLX1NFVCk7CiAgICBtbWlvUmVhZChpbmZvUHRyLT5oTU1pbywgaW5mb1B0ci0+aW5kYXRhLCBpbmZvUHRyLT5hc2guZHdTdWdnZXN0ZWRCdWZmZXJTaXplKTsKICAgIAogICAgaWYgKGluZm9QdHItPmhpYyAmJgoJKGluZm9QdHItPmZuSUNEZWNvbXByZXNzKShpbmZvUHRyLT5oaWMsIDAsIGluZm9QdHItPmluYmloLCBpbmZvUHRyLT5pbmRhdGEsIAoJCQkJICBpbmZvUHRyLT5vdXRiaWgsIGluZm9QdHItPm91dGRhdGEpICE9IElDRVJSX09LKSB7CglMZWF2ZUNyaXRpY2FsU2VjdGlvbigmaW5mb1B0ci0+Y3MpOwoJV0FSTigiRGVjb21wcmVzc2lvbiBlcnJvclxuIik7CglyZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaWYgKChoREMgPSBHZXREQyhpbmZvUHRyLT5oV25kKSkgIT0gMCkgewoJQU5JTUFURV9QYWludEZyYW1lKGluZm9QdHIsIGhEQyk7CglSZWxlYXNlREMoaW5mb1B0ci0+aFduZCwgaERDKTsKICAgIH0KCiAgICBpZiAoaW5mb1B0ci0+Y3VyckZyYW1lKysgPj0gaW5mb1B0ci0+blRvRnJhbWUpIHsKCWluZm9QdHItPmN1cnJGcmFtZSA9IGluZm9QdHItPm5Gcm9tRnJhbWU7CglpZiAoaW5mb1B0ci0+bkxvb3AgIT0gLTEpIHsKCSAgICBpZiAoLS1pbmZvUHRyLT5uTG9vcCA9PSAwKSB7CgkJQU5JTUFURV9Eb1N0b3AoaW5mb1B0cik7CgkgICAgfQoJfQogICAgfQogICAgTGVhdmVDcml0aWNhbFNlY3Rpb24oJmluZm9QdHItPmNzKTsKCiAgICByZXR1cm4gVFJVRTsKfQoKc3RhdGljIHZvaWQgQ0FMTEJBQ0sgQU5JTUFURV9TZXJ2aWNlQ2FsbGJhY2soVUxPTkdfUFRSIHB0cl8pCnsKICAgIEFOSU1BVEVfSU5GTyoJaW5mb1B0ciA9IChBTklNQVRFX0lORk8qKXB0cl87CgogICAgRW50ZXJDcml0aWNhbFNlY3Rpb24oJmluZm9QdHItPmNzKTsKICAgIEFOSU1BVEVfRHJhd0ZyYW1lKGluZm9QdHIpOwogICAgTGVhdmVDcml0aWNhbFNlY3Rpb24oJmluZm9QdHItPmNzKTsKfQoKc3RhdGljIExSRVNVTFQgQU5JTUFURV9QbGF5KEhXTkQgaFduZCwgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgQU5JTUFURV9JTkZPICppbmZvUHRyID0gQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpOwoKICAgIC8qIG5vdGhpbmcgb3BlbmVkICovCiAgICBpZiAoIWluZm9QdHItPmhNTWlvKQoJcmV0dXJuIEZBTFNFOwoKICAgIGlmIChpbmZvUHRyLT5oU2VydmljZSB8fCBpbmZvUHRyLT51VGltZXIpIHsKCUZJWE1FKCJBbHJlYWR5IHBsYXlpbmcgPyB3aGF0IHNob3VsZCBJIGRvID8/XG4iKTsKCUFOSU1BVEVfRG9TdG9wKGluZm9QdHIpOwogICAgfQoKICAgIGluZm9QdHItPm5Gcm9tRnJhbWUgPSAoSU5UKUxPV09SRChsUGFyYW0pOwogICAgaW5mb1B0ci0+blRvRnJhbWUgICA9IChJTlQpSElXT1JEKGxQYXJhbSk7CiAgICBpbmZvUHRyLT5uTG9vcCAgICAgID0gKElOVCl3UGFyYW07CgogICAgaWYgKGluZm9QdHItPm5Ub0ZyYW1lID09IDB4RkZGRikKCWluZm9QdHItPm5Ub0ZyYW1lID0gaW5mb1B0ci0+bWFoLmR3VG90YWxGcmFtZXMgLSAxOwoKICAgIFRSQUNFKCIocmVwZWF0PSVkIGZyb209JWQgdG89JWQpO1xuIiwgCgkgIGluZm9QdHItPm5Mb29wLCBpbmZvUHRyLT5uRnJvbUZyYW1lLCBpbmZvUHRyLT5uVG9GcmFtZSk7CgogICAgaWYgKGluZm9QdHItPm5Gcm9tRnJhbWUgPj0gaW5mb1B0ci0+blRvRnJhbWUgfHwKCWluZm9QdHItPm5Ub0ZyYW1lID49IGluZm9QdHItPm1haC5kd1RvdGFsRnJhbWVzKQoJcmV0dXJuIEZBTFNFOwoKICAgIGluZm9QdHItPmN1cnJGcmFtZSA9IGluZm9QdHItPm5Gcm9tRnJhbWU7CgogICAgaWYgKEdldFdpbmRvd0xvbmdBKGhXbmQsIEdXTF9TVFlMRSkgJiBBQ1NfVElNRVIpIHsKCVRSQUNFKCJVc2luZyBhIHRpbWVyXG4iKTsKCS8qIGNyZWF0ZSBhIHRpbWVyIHRvIGRpc3BsYXkgQVZJICovCglpbmZvUHRyLT51VGltZXIgPSBTZXRUaW1lcihoV25kLCAxLCBpbmZvUHRyLT5tYWguZHdNaWNyb1NlY1BlckZyYW1lIC8gMTAwMCwgTlVMTCk7CiAgICB9IGVsc2UgewoJVFJBQ0UoIlVzaW5nIHRoZSBzZXJ2aWNlIHRocmVhZFxuIik7CgkvKiB0aW1lIGlzIGluILVzICovCglpbmZvUHRyLT5oU2VydmljZSA9IFNFUlZJQ0VfQWRkVGltZXIoaW5mb1B0ci0+bWFoLmR3TWljcm9TZWNQZXJGcmFtZSwgCgkJCQkJICAgICBBTklNQVRFX1NlcnZpY2VDYWxsYmFjaywgKERXT1JEKWluZm9QdHIpOwogICAgfQoJCiAgICBBTklNQVRFX05vdGlmeShpbmZvUHRyLCBBQ05fU1RBUlQpOwoKICAgIHJldHVybiBUUlVFOwp9CgoKc3RhdGljIEJPT0wgQU5JTUFURV9HZXRBdmlJbmZvKEFOSU1BVEVfSU5GTyAqaW5mb1B0cikKewogICAgTU1DS0lORk8JCWNrTWFpblJJRkY7CiAgICBNTUNLSU5GTwkJbW1ja0hlYWQ7CiAgICBNTUNLSU5GTwkJbW1ja0xpc3Q7CiAgICBNTUNLSU5GTwkJbW1ja0luZm87CiAgICBEV09SRAkJbnVtRnJhbWU7CiAgICBEV09SRAkJaW5zaXplOwoKICAgIGlmIChtbWlvRGVzY2VuZChpbmZvUHRyLT5oTU1pbywgJmNrTWFpblJJRkYsIE5VTEwsIDApICE9IDApIHsKCVdBUk4oIkNhbid0IGZpbmQgJ1JJRkYnIGNodW5rXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAoKGNrTWFpblJJRkYuY2tpZCAhPSBGT1VSQ0NfUklGRikgfHwKCShja01haW5SSUZGLmZjY1R5cGUgIT0gbW1pb0ZPVVJDQygnQScsICdWJywgJ0knLCAnICcpKSkgewoJV0FSTigiQ2FuJ3QgZmluZCAnQVZJICcgY2h1bmtcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIG1tY2tIZWFkLmZjY1R5cGUgPSBtbWlvRk9VUkNDKCdoJywgJ2QnLCAncicsICdsJyk7CiAgICBpZiAobW1pb0Rlc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrSGVhZCwgJmNrTWFpblJJRkYsIE1NSU9fRklORExJU1QpICE9IDApIHsKCVdBUk4oIkNhbid0IGZpbmQgJ2hkcmwnIGxpc3RcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIG1tY2tJbmZvLmNraWQgPSBtbWlvRk9VUkNDKCdhJywgJ3YnLCAnaScsICdoJyk7CiAgICBpZiAobW1pb0Rlc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrSW5mbywgJm1tY2tIZWFkLCBNTUlPX0ZJTkRDSFVOSykgIT0gMCkgewoJV0FSTigiQ2FuJ3QgZmluZCAnYXZpaCcgY2h1bmtcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIG1taW9SZWFkKGluZm9QdHItPmhNTWlvLCAoTFBTVFIpJmluZm9QdHItPm1haCwgc2l6ZW9mKGluZm9QdHItPm1haCkpOwogICAgVFJBQ0UoIm1haC5kd01pY3JvU2VjUGVyRnJhbWU9JWxkXG4iLCAJaW5mb1B0ci0+bWFoLmR3TWljcm9TZWNQZXJGcmFtZSk7CiAgICBUUkFDRSgibWFoLmR3TWF4Qnl0ZXNQZXJTZWM9JWxkXG4iLCAJaW5mb1B0ci0+bWFoLmR3TWF4Qnl0ZXNQZXJTZWMpOwogICAgVFJBQ0UoIm1haC5kd1BhZGRpbmdHcmFudWxhcml0eT0lbGRcbiIsIAlpbmZvUHRyLT5tYWguZHdQYWRkaW5nR3JhbnVsYXJpdHkpOwogICAgVFJBQ0UoIm1haC5kd0ZsYWdzPSVsZFxuIiwgCQkJaW5mb1B0ci0+bWFoLmR3RmxhZ3MpOwogICAgVFJBQ0UoIm1haC5kd1RvdGFsRnJhbWVzPSVsZFxuIiwgCQlpbmZvUHRyLT5tYWguZHdUb3RhbEZyYW1lcyk7CiAgICBUUkFDRSgibWFoLmR3SW5pdGlhbEZyYW1lcz0lbGRcbiIsIAkJaW5mb1B0ci0+bWFoLmR3SW5pdGlhbEZyYW1lcyk7CiAgICBUUkFDRSgibWFoLmR3U3RyZWFtcz0lbGRcbiIsIAkJaW5mb1B0ci0+bWFoLmR3U3RyZWFtcyk7CiAgICBUUkFDRSgibWFoLmR3U3VnZ2VzdGVkQnVmZmVyU2l6ZT0lbGRcbiIsCWluZm9QdHItPm1haC5kd1N1Z2dlc3RlZEJ1ZmZlclNpemUpOwogICAgVFJBQ0UoIm1haC5kd1dpZHRoPSVsZFxuIiwgCQkJaW5mb1B0ci0+bWFoLmR3V2lkdGgpOwogICAgVFJBQ0UoIm1haC5kd0hlaWdodD0lbGRcbiIsIAkJaW5mb1B0ci0+bWFoLmR3SGVpZ2h0KTsKICAgIG1taW9Bc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrSW5mbywgMCk7CgogICAgbW1ja0xpc3QuZmNjVHlwZSA9IG1taW9GT1VSQ0MoJ3MnLCAndCcsICdyJywgJ2wnKTsKICAgIGlmIChtbWlvRGVzY2VuZChpbmZvUHRyLT5oTU1pbywgJm1tY2tMaXN0LCAmbW1ja0hlYWQsIE1NSU9fRklORExJU1QpICE9IDApIHsKCVdBUk4oIkNhbid0IGZpbmQgJ3N0cmwnIGxpc3RcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIG1tY2tJbmZvLmNraWQgPSBtbWlvRk9VUkNDKCdzJywgJ3QnLCAncicsICdoJyk7CiAgICBpZiAobW1pb0Rlc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrSW5mbywgJm1tY2tMaXN0LCBNTUlPX0ZJTkRDSFVOSykgIT0gMCkgewoJV0FSTigiQ2FuJ3QgZmluZCAnc3RyaCcgY2h1bmtcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIG1taW9SZWFkKGluZm9QdHItPmhNTWlvLCAoTFBTVFIpJmluZm9QdHItPmFzaCwgc2l6ZW9mKGluZm9QdHItPmFzaCkpOwogICAgVFJBQ0UoImFzaC5mY2NUeXBlPSclYyVjJWMlYydcbiIsIAkJTE9CWVRFKExPV09SRChpbmZvUHRyLT5hc2guZmNjVHlwZSkpLCAKCSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBISUJZVEUoTE9XT1JEKGluZm9QdHItPmFzaC5mY2NUeXBlKSksIAoJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIExPQllURShISVdPUkQoaW5mb1B0ci0+YXNoLmZjY1R5cGUpKSwgCgkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgSElCWVRFKEhJV09SRChpbmZvUHRyLT5hc2guZmNjVHlwZSkpKTsKICAgIFRSQUNFKCJhc2guZmNjSGFuZGxlcj0nJWMlYyVjJWMnXG4iLAlMT0JZVEUoTE9XT1JEKGluZm9QdHItPmFzaC5mY2NIYW5kbGVyKSksIAoJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEhJQllURShMT1dPUkQoaW5mb1B0ci0+YXNoLmZjY0hhbmRsZXIpKSwgCgkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgTE9CWVRFKEhJV09SRChpbmZvUHRyLT5hc2guZmNjSGFuZGxlcikpLCAKCSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBISUJZVEUoSElXT1JEKGluZm9QdHItPmFzaC5mY2NIYW5kbGVyKSkpOwogICAgVFJBQ0UoImFzaC5kd0ZsYWdzPSVsZFxuIiwgCQkJaW5mb1B0ci0+YXNoLmR3RmxhZ3MpOwogICAgVFJBQ0UoImFzaC53UHJpb3JpdHk9JWRcbiIsIAkJaW5mb1B0ci0+YXNoLndQcmlvcml0eSk7CiAgICBUUkFDRSgiYXNoLndMYW5ndWFnZT0lZFxuIiwgCQlpbmZvUHRyLT5hc2gud0xhbmd1YWdlKTsKICAgIFRSQUNFKCJhc2guZHdJbml0aWFsRnJhbWVzPSVsZFxuIiwgCQlpbmZvUHRyLT5hc2guZHdJbml0aWFsRnJhbWVzKTsKICAgIFRSQUNFKCJhc2guZHdTY2FsZT0lbGRcbiIsIAkJCWluZm9QdHItPmFzaC5kd1NjYWxlKTsKICAgIFRSQUNFKCJhc2guZHdSYXRlPSVsZFxuIiwgCQkJaW5mb1B0ci0+YXNoLmR3UmF0ZSk7CiAgICBUUkFDRSgiYXNoLmR3U3RhcnQ9JWxkXG4iLCAJCQlpbmZvUHRyLT5hc2guZHdTdGFydCk7CiAgICBUUkFDRSgiYXNoLmR3TGVuZ3RoPSVsZFxuIiwgCQlpbmZvUHRyLT5hc2guZHdMZW5ndGgpOwogICAgVFJBQ0UoImFzaC5kd1N1Z2dlc3RlZEJ1ZmZlclNpemU9JWxkXG4iLCAJaW5mb1B0ci0+YXNoLmR3U3VnZ2VzdGVkQnVmZmVyU2l6ZSk7CiAgICBUUkFDRSgiYXNoLmR3UXVhbGl0eT0lbGRcbiIsIAkJaW5mb1B0ci0+YXNoLmR3UXVhbGl0eSk7CiAgICBUUkFDRSgiYXNoLmR3U2FtcGxlU2l6ZT0lbGRcbiIsIAkJaW5mb1B0ci0+YXNoLmR3U2FtcGxlU2l6ZSk7CiAgICBUUkFDRSgiYXNoLnJjRnJhbWU9KCVkLCVkLCVkLCVkKVxuIiwgCWluZm9QdHItPmFzaC5yY0ZyYW1lLnRvcCwgaW5mb1B0ci0+YXNoLnJjRnJhbWUubGVmdCwgCgkgIGluZm9QdHItPmFzaC5yY0ZyYW1lLmJvdHRvbSwgaW5mb1B0ci0+YXNoLnJjRnJhbWUucmlnaHQpOwogICAgbW1pb0FzY2VuZChpbmZvUHRyLT5oTU1pbywgJm1tY2tJbmZvLCAwKTsKCiAgICBtbWNrSW5mby5ja2lkID0gbW1pb0ZPVVJDQygncycsICd0JywgJ3InLCAnZicpOwogICAgaWYgKG1taW9EZXNjZW5kKGluZm9QdHItPmhNTWlvLCAmbW1ja0luZm8sICZtbWNrTGlzdCwgTU1JT19GSU5EQ0hVTkspICE9IDApIHsKCVdBUk4oIkNhbid0IGZpbmQgJ3N0cmgnIGNodW5rXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpbmZvUHRyLT5pbmJpaCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBtbWNrSW5mby5ja3NpemUpOwogICAgaWYgKCFpbmZvUHRyLT5pbmJpaCkgewoJV0FSTigiQ2FuJ3QgYWxsb2MgaW5wdXQgQklIXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBtbWlvUmVhZChpbmZvUHRyLT5oTU1pbywgKExQU1RSKWluZm9QdHItPmluYmloLCBtbWNrSW5mby5ja3NpemUpOwogICAgVFJBQ0UoImJpaC5iaVNpemU9JWxkXG4iLCAJCWluZm9QdHItPmluYmloLT5iaVNpemUpOwogICAgVFJBQ0UoImJpaC5iaVdpZHRoPSVsZFxuIiwgCQlpbmZvUHRyLT5pbmJpaC0+YmlXaWR0aCk7CiAgICBUUkFDRSgiYmloLmJpSGVpZ2h0PSVsZFxuIiwgCWluZm9QdHItPmluYmloLT5iaUhlaWdodCk7CiAgICBUUkFDRSgiYmloLmJpUGxhbmVzPSVkXG4iLCAJCWluZm9QdHItPmluYmloLT5iaVBsYW5lcyk7CiAgICBUUkFDRSgiYmloLmJpQml0Q291bnQ9JWRcbiIsIAlpbmZvUHRyLT5pbmJpaC0+YmlCaXRDb3VudCk7CiAgICBUUkFDRSgiYmloLmJpQ29tcHJlc3Npb249JWxkXG4iLCAJaW5mb1B0ci0+aW5iaWgtPmJpQ29tcHJlc3Npb24pOwogICAgVFJBQ0UoImJpaC5iaVNpemVJbWFnZT0lbGRcbiIsIAlpbmZvUHRyLT5pbmJpaC0+YmlTaXplSW1hZ2UpOwogICAgVFJBQ0UoImJpaC5iaVhQZWxzUGVyTWV0ZXI9JWxkXG4iLCAJaW5mb1B0ci0+aW5iaWgtPmJpWFBlbHNQZXJNZXRlcik7CiAgICBUUkFDRSgiYmloLmJpWVBlbHNQZXJNZXRlcj0lbGRcbiIsIAlpbmZvUHRyLT5pbmJpaC0+YmlZUGVsc1Blck1ldGVyKTsKICAgIFRSQUNFKCJiaWguYmlDbHJVc2VkPSVsZFxuIiwgCWluZm9QdHItPmluYmloLT5iaUNsclVzZWQpOwogICAgVFJBQ0UoImJpaC5iaUNsckltcG9ydGFudD0lbGRcbiIsIAlpbmZvUHRyLT5pbmJpaC0+YmlDbHJJbXBvcnRhbnQpOwogICAgbW1pb0FzY2VuZChpbmZvUHRyLT5oTU1pbywgJm1tY2tJbmZvLCAwKTsKCiAgICBtbWlvQXNjZW5kKGluZm9QdHItPmhNTWlvLCAmbW1ja0xpc3QsIDApOwogICAgCiNpZiAwCiAgICAvKiBhbiBBVkkgaGFzIDAgb3IgMSB2aWRlbyBzdHJlYW0sIGFuZCB0byBiZSBhbmltYXRlZCBzaG91bGQgbm90IGNvbnRhaW4KICAgICAqIGFuIGF1ZGlvIHN0cmVhbSwgc28gb25seSBvbmUgc3RybCBpcyBhbGxvd2VkIAogICAgICovCiAgICBtbWNrTGlzdC5mY2NUeXBlID0gbW1pb0ZPVVJDQygncycsICd0JywgJ3InLCAnbCcpOwogICAgaWYgKG1taW9EZXNjZW5kKGluZm9QdHItPmhNTWlvLCAmbW1ja0xpc3QsICZtbWNrSGVhZCwgTU1JT19GSU5ETElTVCkgPT0gMCkgewoJV0FSTigiVGhlcmUgc2hvdWxkIGJlIGEgc2luZ2xlICdzdHJsJyBsaXN0XG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KI2VuZGlmCgogICAgbW1pb0FzY2VuZChpbmZvUHRyLT5oTU1pbywgJm1tY2tIZWFkLCAwKTsKCiAgICAvKiBubyBuZWVkIHRvIHJlYWQgb3B0aW9uYWwgSlVOSyBjaHVuayAqLwoKICAgIG1tY2tMaXN0LmZjY1R5cGUgPSBtbWlvRk9VUkNDKCdtJywgJ28nLCAndicsICdpJyk7CiAgICBpZiAobW1pb0Rlc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrTGlzdCwgJmNrTWFpblJJRkYsIE1NSU9fRklORExJU1QpICE9IDApIHsKCVdBUk4oIkNhbid0IGZpbmQgJ21vdmknIGxpc3RcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIC8qIEZJWE1FOiBzaG91bGQgaGFuZGxlIHRoZSAncmVjICcgTElTVCB3aGVuIHByZXNlbnQgKi8KCiAgICBpbmZvUHRyLT5scEluZGV4ID0gSGVhcEFsbG9jKEdldFByb2Nlc3NIZWFwKCksIEhFQVBfWkVST19NRU1PUlksIAoJCQkJIGluZm9QdHItPm1haC5kd1RvdGFsRnJhbWVzICogc2l6ZW9mKERXT1JEKSk7CiAgICBpZiAoIWluZm9QdHItPmxwSW5kZXgpIHsKCVdBUk4oIkNhbid0IGFsbG9jIGluZGV4IGFycmF5XG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBudW1GcmFtZSA9IGluc2l6ZSA9IDA7CiAgICB3aGlsZSAobW1pb0Rlc2NlbmQoaW5mb1B0ci0+aE1NaW8sICZtbWNrSW5mbywgJm1tY2tMaXN0LCAwKSA9PSAwICYmIAoJICAgbnVtRnJhbWUgPCBpbmZvUHRyLT5tYWguZHdUb3RhbEZyYW1lcykgewoJaW5mb1B0ci0+bHBJbmRleFtudW1GcmFtZV0gPSBtbWNrSW5mby5kd0RhdGFPZmZzZXQ7CglpZiAoaW5zaXplIDwgbW1ja0luZm8uY2tzaXplKQoJICAgIGluc2l6ZSA9IG1tY2tJbmZvLmNrc2l6ZTsKCW51bUZyYW1lKys7CgltbWlvQXNjZW5kKGluZm9QdHItPmhNTWlvLCAmbW1ja0luZm8sIDApOwogICAgfQogICAgaWYgKG51bUZyYW1lICE9IGluZm9QdHItPm1haC5kd1RvdGFsRnJhbWVzKSB7CglXQVJOKCJGb3VuZCAlbGQgZnJhbWVzICgvJWxkKVxuIiwgbnVtRnJhbWUsIGluZm9QdHItPm1haC5kd1RvdGFsRnJhbWVzKTsKCXJldHVybiBGQUxTRTsKICAgIH0KICAgIGlmIChpbnNpemUgPiBpbmZvUHRyLT5hc2guZHdTdWdnZXN0ZWRCdWZmZXJTaXplKSB7CglXQVJOKCJpbnNpemU9JWxkIHN1Z2dlc3RlZFNpemU9JWxkXG4iLCBpbnNpemUsIGluZm9QdHItPmFzaC5kd1N1Z2dlc3RlZEJ1ZmZlclNpemUpOwoJaW5mb1B0ci0+YXNoLmR3U3VnZ2VzdGVkQnVmZmVyU2l6ZSA9IGluc2l6ZTsKICAgIH0KCiAgICBpbmZvUHRyLT5pbmRhdGEgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgaW5mb1B0ci0+YXNoLmR3U3VnZ2VzdGVkQnVmZmVyU2l6ZSk7CiAgICBpZiAoIWluZm9QdHItPmluZGF0YSkgewoJV0FSTigiQ2FuJ3QgYWxsb2MgaW5wdXQgYnVmZmVyXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICByZXR1cm4gVFJVRTsKfQoKCnN0YXRpYyBCT09MICAgIEFOSU1BVEVfR2V0QXZpQ29kZWMoQU5JTUFURV9JTkZPICppbmZvUHRyKQp7CiAgICBEV09SRAlvdXRTaXplOwoKICAgIC8qIGNoZWNrIHVuY29tcHJlc3NlZCBBVkkgKi8KICAgIGlmIChpbmZvUHRyLT5hc2guZmNjSGFuZGxlciA9PSBtbWlvRk9VUkNDKCdEJywgJ0knLCAnQicsICcgJykpIHsKCWluZm9QdHItPmhpYyA9IDA7CglyZXR1cm4gVFJVRTsKICAgIH0KCiAgICAvKiB0cnkgdG8gZ2V0IGEgZGVjb21wcmVzc29yIGZvciB0aGF0IHR5cGUgKi8KICAgIGluZm9QdHItPmhpYyA9IChpbmZvUHRyLT5mbklDT3BlbikoSUNUWVBFX1ZJREVPLCAKCQkJCSAgICAgICBpbmZvUHRyLT5hc2guZmNjSGFuZGxlciwgCgkJCQkgICAgICAgSUNNT0RFX0RFQ09NUFJFU1MpOwogICAgaWYgKCFpbmZvUHRyLT5oaWMpIHsKCVdBUk4oIkNhbid0IGxvYWQgY29kZWMgZm9yIHRoZSBmaWxlXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KICAgIAogICAgb3V0U2l6ZSA9IChpbmZvUHRyLT5mbklDU2VuZE1lc3NhZ2UpKGluZm9QdHItPmhpYywgCgkJCQkJIElDTV9ERUNPTVBSRVNTX0dFVF9GT1JNQVQsIAoJCQkJCSAoRFdPUkQpaW5mb1B0ci0+aW5iaWgsIDBMKTsKICAgIGluZm9QdHItPm91dGJpaCA9IEhlYXBBbGxvYyhHZXRQcm9jZXNzSGVhcCgpLCAwLCBvdXRTaXplKTsKICAgIGlmICghaW5mb1B0ci0+b3V0YmloKSB7CglXQVJOKCJDYW4ndCBhbGxvYyBvdXRwdXQgQklIXG4iKTsKCXJldHVybiBGQUxTRTsKICAgIH0KCiAgICBpZiAoKGluZm9QdHItPmZuSUNTZW5kTWVzc2FnZSkoaW5mb1B0ci0+aGljLCBJQ01fREVDT01QUkVTU19HRVRfRk9STUFULCAKCQkJCSAgIChEV09SRClpbmZvUHRyLT5pbmJpaCwgCgkJCQkgICAoRFdPUkQpaW5mb1B0ci0+b3V0YmloKSAhPSBJQ0VSUl9PSykgewoJV0FSTigiQ2FuJ3QgZ2V0IG91dHB1dCBCSUhcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGluZm9QdHItPm91dGRhdGEgPSBIZWFwQWxsb2MoR2V0UHJvY2Vzc0hlYXAoKSwgMCwgaW5mb1B0ci0+b3V0YmloLT5iaVNpemVJbWFnZSk7CiAgICBpZiAoIWluZm9QdHItPm91dGRhdGEpIHsKCVdBUk4oIkNhbid0IGFsbG9jIG91dHB1dCBidWZmZXJcbiIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmICgoaW5mb1B0ci0+Zm5JQ1NlbmRNZXNzYWdlKShpbmZvUHRyLT5oaWMsIElDTV9ERUNPTVBSRVNTX0JFR0lOLCAKCQkJCSAgIChEV09SRClpbmZvUHRyLT5pbmJpaCwgCgkJCQkgICAoRFdPUkQpaW5mb1B0ci0+b3V0YmloKSAhPSBJQ0VSUl9PSykgewoJV0FSTigiQ2FuJ3QgYmVnaW4gZGVjb21wcmVzc2lvblxuIik7CglyZXR1cm4gRkFMU0U7CiAgICB9CgogICAgcmV0dXJuIFRSVUU7Cn0KCnN0YXRpYyBMUkVTVUxUIEFOSU1BVEVfT3BlbkEoSFdORCBoV25kLCBXUEFSQU0gd1BhcmFtLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBBTklNQVRFX0lORk8gKmluZm9QdHIgPSBBTklNQVRFX0dldEluZm9QdHIoaFduZCk7CiAgICBISU5TVEFOQ0UgaEluc3RhbmNlID0gKEhJTlNUQU5DRSl3UGFyYW07CgogICAgQU5JTUFURV9GcmVlKGluZm9QdHIpOwoKICAgIGlmICghbFBhcmFtKSB7CglUUkFDRSgiQ2xvc2luZyBhdmkhXG4iKTsKCXJldHVybiBUUlVFOwogICAgfQogICAgCiAgICBpZiAoIWhJbnN0YW5jZSkKICAgICAgIGhJbnN0YW5jZSA9IEdldFdpbmRvd0xvbmdBKGhXbmQsIEdXTF9ISU5TVEFOQ0UpOwoKICAgIGlmIChISVdPUkQobFBhcmFtKSkgewoJVFJBQ0UoIihcIiVzXCIpO1xuIiwgKExQU1RSKWxQYXJhbSk7CgoJaWYgKCFBTklNQVRFX0xvYWRSZXNBKGluZm9QdHIsIGhJbnN0YW5jZSwgKExQU1RSKWxQYXJhbSkpIHsKCSAgICBUUkFDRSgiTm8gQVZJIHJlc291cmNlIGZvdW5kIVxuIik7CgkgICAgaWYgKCFBTklNQVRFX0xvYWRGaWxlQShpbmZvUHRyLCAoTFBTVFIpbFBhcmFtKSkgewoJCVdBUk4oIk5vIEFWSSBmaWxlIGZvdW5kIVxuIik7CgkJcmV0dXJuIEZBTFNFOwoJICAgIH0KCX0KICAgIH0gZWxzZSB7CglUUkFDRSgiKCV1KTtcbiIsIChXT1JEKUxPV09SRChsUGFyYW0pKTsKCglpZiAoIUFOSU1BVEVfTG9hZFJlc0EoaW5mb1B0ciwgaEluc3RhbmNlLAoJCQkgICAgICBNQUtFSU5UUkVTT1VSQ0VBKChJTlQpbFBhcmFtKSkpIHsKCSAgICBXQVJOKCJObyBBVkkgcmVzb3VyY2UgZm91bmQhXG4iKTsKCSAgICByZXR1cm4gRkFMU0U7Cgl9CiAgICB9CgogICAgaWYgKCFBTklNQVRFX0dldEF2aUluZm8oaW5mb1B0cikpIHsKCVdBUk4oIkNhbid0IGdldCBBVkkgaW5mb3JtYXRpb25cbiIpOwoJQU5JTUFURV9GcmVlKGluZm9QdHIpOwoJcmV0dXJuIEZBTFNFOwogICAgfQoKICAgIGlmICghQU5JTUFURV9HZXRBdmlDb2RlYyhpbmZvUHRyKSkgewoJV0FSTigiQ2FuJ3QgZ2V0IEFWSSBDb2RlY1xuIik7CglBTklNQVRFX0ZyZWUoaW5mb1B0cik7CglyZXR1cm4gRkFMU0U7CiAgICB9CgogICAgaWYgKEdldFdpbmRvd0xvbmdBKGhXbmQsIEdXTF9TVFlMRSkgJiBBQ1NfQ0VOVEVSKSB7CglGSVhNRSgiQUNTX0NFTlRFUjogTklZXG4iKTsKICAgIH0gZWxzZSB7CgkvKglNb3ZlV2luZG93KGhXbmQsIDAsIDAsIGluZm9QdHItPm1haC5kd1dpZHRoLCBpbmZvUHRyLT5tYWguZHdIZWlnaHQsIEZBTFNFKTsqLwoJU2V0V2luZG93UG9zKGhXbmQsIDAsIDAsIDAsIGluZm9QdHItPm1haC5kd1dpZHRoLCBpbmZvUHRyLT5tYWguZHdIZWlnaHQsCgkJICAgICBTV1BfTk9BQ1RJVkFURSB8IFNXUF9OT01PVkUgfCBTV1BfTk9aT1JERVIpOwogICAgfQoKICAgIGlmIChHZXRXaW5kb3dMb25nQShoV25kLCBHV0xfU1RZTEUpICYgQUNTX1RSQU5TUEFSRU5UKSB7CglGSVhNRSgiQUNTX1RSQU5TUEFSRU5UOiBOSVlcbiIpOwogICAgfQoKICAgIGlmIChHZXRXaW5kb3dMb25nQShoV25kLCBHV0xfU1RZTEUpICYgQUNTX0FVVE9QTEFZKSB7CglyZXR1cm4gQU5JTUFURV9QbGF5KGhXbmQsIC0xLCAoTFBBUkFNKU1BS0VMT05HKDAsIGluZm9QdHItPm1haC5kd1RvdGFsRnJhbWVzLTEpKTsKICAgIH0KCiAgICByZXR1cm4gVFJVRTsKfQoKCi8qIDw8IEFOSU1BVEVfT3BlbjMyVyA+PiAqLwoKc3RhdGljIExSRVNVTFQgQU5JTUFURV9TdG9wKEhXTkQgaFduZCwgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgQU5JTUFURV9JTkZPICppbmZvUHRyID0gQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpOwoKICAgIC8qIG5vdGhpbmcgb3BlbmVkICovCiAgICBpZiAoIWluZm9QdHItPmhNTWlvKQoJcmV0dXJuIEZBTFNFOwoKICAgIEFOSU1BVEVfRG9TdG9wKGluZm9QdHIpOwogICAgcmV0dXJuIFRSVUU7Cn0KCgpzdGF0aWMgTFJFU1VMVCBBTklNQVRFX0NyZWF0ZShIV05EIGhXbmQsIFdQQVJBTSB3UGFyYW0sIExQQVJBTSBsUGFyYW0pCnsKICAgIEFOSU1BVEVfSU5GTyoJaW5mb1B0cjsKICAgIEhNT0RVTEUJCWhNb2R1bGUgPSBMb2FkTGlicmFyeUEoIm1zdmZ3MzIuZGxsIik7CgogICAgaWYgKCFoTW9kdWxlKQoJcmV0dXJuIEZBTFNFOwoKICAgIC8qIGFsbG9jYXRlIG1lbW9yeSBmb3IgaW5mbyBzdHJ1Y3R1cmUgKi8KICAgIGluZm9QdHIgPSAoQU5JTUFURV9JTkZPICopQ09NQ1RMMzJfQWxsb2Moc2l6ZW9mKEFOSU1BVEVfSU5GTykpOwogICAgaWYgKCFpbmZvUHRyKSB7CglFUlIoImNvdWxkIG5vdCBhbGxvY2F0ZSBpbmZvIG1lbW9yeSFcbiIpOwoJcmV0dXJuIDA7CiAgICB9CgogICAgLyogVGVtcG9yYXJ5IGhhY2sgdW50aWwgd2UgZ2V0IGRsbGdsdWUgdXAgYW5kIHJ1bm5pbmcgKi8KICAgIGluZm9QdHItPmZuSUNPcGVuICAgICAgICA9ICh2b2lkKilHZXRQcm9jQWRkcmVzcyhoTW9kdWxlLCAiSUNPcGVuIik7CiAgICBpbmZvUHRyLT5mbklDQ2xvc2UgICAgICAgPSAodm9pZCopR2V0UHJvY0FkZHJlc3MoaE1vZHVsZSwgIklDQ2xvc2UiKTsKICAgIGluZm9QdHItPmZuSUNTZW5kTWVzc2FnZSA9ICh2b2lkKilHZXRQcm9jQWRkcmVzcyhoTW9kdWxlLCAiSUNTZW5kTWVzc2FnZSIpOwogICAgaW5mb1B0ci0+Zm5JQ0RlY29tcHJlc3MgID0gKHZvaWQqKUdldFByb2NBZGRyZXNzKGhNb2R1bGUsICJJQ0RlY29tcHJlc3MiKTsKCiAgICBUUkFDRSgiQW5pbWF0ZSBzdHlsZT0weCUwOGx4LCBwYXJlbnQ9JTA4bHhcbiIsIEdldFdpbmRvd0xvbmdBKGhXbmQsIEdXTF9TVFlMRSksIChEV09SRClHZXRQYXJlbnQoaFduZCkpOwoKICAgIC8qIHN0b3JlIGNyb3NzcmVmIGhXbmQgPC0+IGluZm8gc3RydWN0dXJlICovCiAgICBTZXRXaW5kb3dMb25nQShoV25kLCAwLCAoRFdPUkQpaW5mb1B0cik7CiAgICBpbmZvUHRyLT5oV25kID0gaFduZDsKCiAgICBJbml0aWFsaXplQ3JpdGljYWxTZWN0aW9uKCZpbmZvUHRyLT5jcyk7CiAgICAKICAgIHJldHVybiAwOwp9CgoKc3RhdGljIExSRVNVTFQgQU5JTUFURV9EZXN0cm95KEhXTkQgaFduZCwgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgQU5JTUFURV9JTkZPICppbmZvUHRyID0gQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpOwoKCiAgICAvKiBmcmVlIGF2aSBkYXRhICovCiAgICBBTklNQVRFX0ZyZWUoaW5mb1B0cik7CgogICAgLyogZnJlZSBhbmltYXRlIGluZm8gZGF0YSAqLwogICAgQ09NQ1RMMzJfRnJlZShpbmZvUHRyKTsKCiAgICByZXR1cm4gMDsKfQoKCnN0YXRpYyBMUkVTVUxUIEFOSU1BVEVfRXJhc2VCYWNrZ3JvdW5kKEhXTkQgaFduZCwgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgUkVDVCByZWN0OwoKICAgIEdldENsaWVudFJlY3QoaFduZCwgJnJlY3QpOwojaWYgMAogICAgSEJSVVNIIGhCcnVzaCA9IENyZWF0ZVNvbGlkQnJ1c2goaW5mb1B0ci0+Y2xyQmspOwoKICAgIEZpbGxSZWN0KChIREMpd1BhcmFtLCAmcmVjdCwgaEJydXNoKTsKICAgIERlbGV0ZU9iamVjdChoQnJ1c2gpOwojZWxzZQogICAgRmlsbFJlY3QoKEhEQyl3UGFyYW0sICZyZWN0LCBHZXRTeXNDb2xvckJydXNoKENPTE9SX1dJTkRPVykpOwojZW5kaWYKICAgIHJldHVybiBUUlVFOwp9CgpzdGF0aWMgTFJFU1VMVCBXSU5BUEkgQU5JTUFURV9TaXplKEhXTkQgaFduZCwgV1BBUkFNIHdQYXJhbSwgTFBBUkFNIGxQYXJhbSkKewogICAgQU5JTUFURV9JTkZPICppbmZvUHRyID0gQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpOwoKICAgIGlmIChHZXRXaW5kb3dMb25nQShoV25kLCBHV0xfU1RZTEUpICYgQUNTX0NFTlRFUikgewoJRklYTUUoIk5JWVxuIik7CglpZiAoaW5mb1B0ci0+aE1NaW8pIHsKCSAgICAvKiBjZW50ZXJzIHRoZSBhbmltYXRpb24gaW4gdGhlIGNvbnRyb2wsIGludmFsaWRhdGVzIHRoZSBjb250cm9sCgkgICAgICovCgl9CglJbnZhbGlkYXRlUmVjdChoV25kLCBOVUxMLCBUUlVFKTsKICAgIH0KICAgIHJldHVybiBUUlVFOwp9CgpzdGF0aWMgTFJFU1VMVCBXSU5BUEkgQU5JTUFURV9XaW5kb3dQcm9jKEhXTkQgaFduZCwgVUlOVCB1TXNnLCBXUEFSQU0gd1BhcmFtLCBMUEFSQU0gbFBhcmFtKQp7CiAgICBzd2l0Y2ggKHVNc2cpCiAgICB7CiAgICBjYXNlIEFDTV9PUEVOQToKCXJldHVybiBBTklNQVRFX09wZW5BKGhXbmQsIHdQYXJhbSwgbFBhcmFtKTsKCQoJLyoJY2FzZSBBQ01fT1BFTjMyVzogRklYTUUhISAqLwoJLyoJICAgIHJldHVybiBBTklNQVRFX09wZW4zMlcoaFduZCwgd1BhcmFtLCBsUGFyYW0pOyAqLwoJCiAgICBjYXNlIEFDTV9QTEFZOgoJcmV0dXJuIEFOSU1BVEVfUGxheShoV25kLCB3UGFyYW0sIGxQYXJhbSk7CgkKICAgIGNhc2UgQUNNX1NUT1A6CglyZXR1cm4gQU5JTUFURV9TdG9wKGhXbmQsIHdQYXJhbSwgbFBhcmFtKTsKCQogICAgY2FzZSBXTV9OQ0NSRUFURToKCUFOSU1BVEVfQ3JlYXRlKGhXbmQsIHdQYXJhbSwgbFBhcmFtKTsKCXJldHVybiBEZWZXaW5kb3dQcm9jQShoV25kLCB1TXNnLCB3UGFyYW0sIGxQYXJhbSk7CgkKICAgIGNhc2UgV01fTkNISVRURVNUOgoJcmV0dXJuIEhUVFJBTlNQQVJFTlQ7CgogICAgY2FzZSBXTV9ERVNUUk9ZOgoJQU5JTUFURV9EZXN0cm95KGhXbmQsIHdQYXJhbSwgbFBhcmFtKTsKCXJldHVybiBEZWZXaW5kb3dQcm9jQShoV25kLCB1TXNnLCB3UGFyYW0sIGxQYXJhbSk7CgkKICAgIGNhc2UgV01fRVJBU0VCS0dORDoKCUFOSU1BVEVfRXJhc2VCYWNrZ3JvdW5kKGhXbmQsIHdQYXJhbSwgbFBhcmFtKTsKCWJyZWFrOwoKICAgIC8qCWNhc2UgV01fU1RZTEVDSEFOR0VEOiBGSVhNRSBzaGFsbCB3ZSBkbyBzb21ldGhpbmcgPz8gKi8KCiAgICBjYXNlIFdNX1RJTUVSOgoJcmV0dXJuIEFOSU1BVEVfRHJhd0ZyYW1lKEFOSU1BVEVfR2V0SW5mb1B0cihoV25kKSk7CgkKICAgIGNhc2UgV01fQ0xPU0U6CglBTklNQVRFX0ZyZWUoQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpKTsKCXJldHVybiBUUlVFOwoKICAgIGNhc2UgV01fUEFJTlQ6CglpZiAod1BhcmFtKSB7CgkgICAgQU5JTUFURV9QYWludEZyYW1lKEFOSU1BVEVfR2V0SW5mb1B0cihoV25kKSwgKEhEQyl3UGFyYW0pOwoJfSBlbHNlIHsKCSAgICBQQUlOVFNUUlVDVCBwczsKIAkgICAgSERDIGhEQyA9IEJlZ2luUGFpbnQoaFduZCwgJnBzKTsKCSAgICBBTklNQVRFX1BhaW50RnJhbWUoQU5JTUFURV9HZXRJbmZvUHRyKGhXbmQpLCBoREMpOwoJICAgIEVuZFBhaW50KGhXbmQsICZwcyk7Cgl9CglicmVhazsKCiAgICBjYXNlIFdNX1NJWkU6CglBTklNQVRFX1NpemUoaFduZCwgd1BhcmFtLCBsUGFyYW0pOwoJcmV0dXJuIERlZldpbmRvd1Byb2NBKGhXbmQsIHVNc2csIHdQYXJhbSwgbFBhcmFtKTsKCiAgICBkZWZhdWx0OgoJaWYgKHVNc2cgPj0gV01fVVNFUikKCSAgICBFUlIoInVua25vd24gbXNnICUwNHggd3A9JTA4eCBscD0lMDhseFxuIiwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwoJCglyZXR1cm4gRGVmV2luZG93UHJvY0EoaFduZCwgdU1zZywgd1BhcmFtLCBsUGFyYW0pOwogICAgfQogICAgcmV0dXJuIDA7Cn0KCgp2b2lkIEFOSU1BVEVfUmVnaXN0ZXIodm9pZCkKewogICAgV05EQ0xBU1NBIHduZENsYXNzOwoKICAgIFplcm9NZW1vcnkoJnduZENsYXNzLCBzaXplb2YoV05EQ0xBU1NBKSk7CiAgICB3bmRDbGFzcy5zdHlsZSAgICAgICAgID0gQ1NfR0xPQkFMQ0xBU1MgfCBDU19EQkxDTEtTOwogICAgd25kQ2xhc3MubHBmblduZFByb2MgICA9IChXTkRQUk9DKUFOSU1BVEVfV2luZG93UHJvYzsKICAgIHduZENsYXNzLmNiQ2xzRXh0cmEgICAgPSAwOwogICAgd25kQ2xhc3MuY2JXbmRFeHRyYSAgICA9IHNpemVvZihBTklNQVRFX0lORk8gKik7CiAgICB3bmRDbGFzcy5oQ3Vyc29yICAgICAgID0gTG9hZEN1cnNvckEoMCwgSURDX0FSUk9XQSk7CiAgICB3bmRDbGFzcy5oYnJCYWNrZ3JvdW5kID0gKEhCUlVTSCkoQ09MT1JfQlRORkFDRSArIDEpOwogICAgd25kQ2xhc3MubHBzekNsYXNzTmFtZSA9IEFOSU1BVEVfQ0xBU1NBOwogCiAgICBSZWdpc3RlckNsYXNzQSgmd25kQ2xhc3MpOwp9CgoKdm9pZCBBTklNQVRFX1VucmVnaXN0ZXIodm9pZCkKewogICAgVW5yZWdpc3RlckNsYXNzQShBTklNQVRFX0NMQVNTQSwgKEhJTlNUQU5DRSlOVUxMKTsKfQoK