LyoKKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiogQ29weXJpZ2h0IChDKSB7MTk5Ni0yMDAxfSwgSW50ZXJuYXRpb25hbCBCdXNpbmVzcyBNYWNoaW5lcyBDb3Jwb3JhdGlvbiBhbmQgICAqCiogb3RoZXJzLiBBbGwgUmlnaHRzIFJlc2VydmVkLiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKgoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKKi8KCi8qKgoqIEZpbGUgdGJsY29sbC5jcHAKKgoqIENyZWF0ZWQgYnk6IEhlbGVuYSBTaGloCioKKiBNb2RpZmljYXRpb24gSGlzdG9yeToKKgoqICBEYXRlICAgICAgICBOYW1lICAgICAgICBEZXNjcmlwdGlvbgoqICAyLzUvOTcgICAgICBhbGl1ICAgICAgICBBZGRlZCBzdHJlYW1JbiBhbmQgc3RyZWFtT3V0IG1ldGhvZHMuICBBZGRlZAoqICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdHJ1Y3RvciB3aGljaCByZWFkcyBSdWxlQmFzZWRDb2xsYXRvciBvYmplY3QgZnJvbQoqICAgICAgICAgICAgICAgICAgICAgICAgICBhIGJpbmFyeSBmaWxlLiAgQWRkZWQgd3JpdGVUb0ZpbGUgbWV0aG9kIHdoaWNoIHN0cmVhbXMKKiAgICAgICAgICAgICAgICAgICAgICAgICAgUnVsZUJhc2VkQ29sbGF0b3Igb3V0IHRvIGEgYmluYXJ5IGZpbGUuICBUaGUgc3RyZWFtSW4KKiAgICAgICAgICAgICAgICAgICAgICAgICAgYW5kIHN0cmVhbU91dCBtZXRob2RzIHVzZSBpc3RyZWFtIGFuZCBvc3RyZWFtIG9iamVjdHMKKiAgICAgICAgICAgICAgICAgICAgICAgICAgaW4gYmluYXJ5IG1vZGUuCiogIDIvMTEvOTcgICAgIGFsaXUgICAgICAgIE1vdmVkIGRlY2xhcmF0aW9ucyBvdXQgb2YgZm9yIGxvb3AgaW5pdGlhbGl6ZXIuCiogICAgICAgICAgICAgICAgICAgICAgICAgIEFkZGVkIE1hYyBjb21wYXRpYmlsaXR5ICNpZmRlZiBmb3IgaW9zOjpub2NyZWF0ZS4KKiAgMi8xMi85NyAgICAgYWxpdSAgICAgICAgTW9kaWZpZWQgdG8gdXNlIFRhYmxlQ29sbGF0aW9uRGF0YSBzdWItb2JqZWN0IHRvCiogICAgICAgICAgICAgICAgICAgICAgICAgIGhvbGQgaW52YXJpYW50IGRhdGEuCiogIDIvMTMvOTcgICAgIGFsaXUgICAgICAgIE1vdmVkIHNldmVyYWwgbWV0aG9kcyBpbnRvIHRoaXMgY2xhc3MgZnJvbSBDb2xsYXRpb24uCiogICAgICAgICAgICAgICAgICAgICAgICAgIEFkZGVkIGEgcHJpdmF0ZSBSdWxlQmFzZWRDb2xsYXRvcihMb2NhbGUmKSBjb25zdHJ1Y3RvciwKKiAgICAgICAgICAgICAgICAgICAgICAgICAgdG8gYmUgdXNlZCBieSBDb2xsYXRvcjo6Z2V0SW5zdGFuY2UoKS4gIEdlbmVyYWwKKiAgICAgICAgICAgICAgICAgICAgICAgICAgY2xlYW4gdXAuICBNYWRlIHVzZSBvZiBVRXJyb3JDb2RlIHZhcmlhYmxlcyBjb25zaXN0ZW50LgoqICAyLzIwLzk3ICAgICBoZWxlbmEgICAgICBBZGRlZCBjbG9uZSwgb3BlcmF0b3I9PSwgb3BlcmF0b3IhPSwgb3BlcmF0b3I9LCBhbmQgY29weQoqICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdHJ1Y3RvciBhbmQgZ2V0RHluYW1pY0NsYXNzSUQuCiogIDMvNS85NyAgICAgIGFsaXUgICAgICAgIENoYW5nZWQgY29tcGFjdGlvbiBjeWNsZSB0byBpbXByb3ZlIHBlcmZvcm1hbmNlLiAgV2UKKiAgICAgICAgICAgICAgICAgICAgICAgICAgdXNlIHRoZSBtYXhpbXVtIGFsbG93YWJsZSB2YWx1ZSB3aGljaCBpcyBrQmxvY2tDb3VudC4KKiAgICAgICAgICAgICAgICAgICAgICAgICAgTW9kaWZpZWQgZ2V0UnVsZXMoKSB0byBsb2FkIHJ1bGVzIGR5bmFtaWNhbGx5LiAgQ2hhbmdlZAoqICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdHJ1Y3RGcm9tRmlsZSgpIGNhbGwgdG8gYWNjb21vZGF0ZSB0aGlzIChhZGRlZAoqICAgICAgICAgICAgICAgICAgICAgICAgICBwYXJhbWV0ZXIgdG8gc3BlY2lmeSB3aGV0aGVyIGJpbmFyeSBsb2FkaW5nIGlzIHRvCiogICAgICAgICAgICAgICAgICAgICAgICAgIHRha2UgcGxhY2UpLgoqIDA1LzA2Lzk3ICAgICBoZWxlbmEgICAgICBBZGRlZCBtZW1vcnkgYWxsb2NhdGlvbiBlcnJvciBjaGVjay4KKiAgNi8yMC85NyAgICAgaGVsZW5hICAgICAgSmF2YSBjbGFzcyBuYW1lIGNoYW5nZS4KKiAgNi8yMy85NyAgICAgaGVsZW5hICAgICAgQWRkaW5nIGNvbW1lbnRzIHRvIG1ha2UgY29kZSBtb3JlIHJlYWRhYmxlLgoqIDA5LzAzLzk3ICAgICBoZWxlbmEgICAgICBBZGRlZCBjcmVhdGVDb2xsYXRpb25LZXlWYWx1ZXMoKS4KKiAwNi8yNi85OCAgICAgZXJtICAgICAgICAgQ2hhbmdlcyBmb3IgQ29sbGF0aW9uS2V5cyB1c2luZyBieXRlIGFycmF5cy4KKiAwOC8xMC85OCAgICAgZXJtICAgICAgICAgU3luY2hlZCB3aXRoIDEuMiB2ZXJzaW9uIG9mIFJ1bGVCYXNlZENvbGxhdG9yLmphdmEKKiAwNC8yMy85OSAgICAgc3RlcGhlbiAgICAgUmVtb3ZlZCBFRGVjb21wb3NpdGlvbk1vZGUsIG1lcmdlZCB3aXRoCiogICAgICAgICAgICAgICAgICAgICAgICAgIE5vcm1hbGl6ZXI6OkVNb2RlCiogMDYvMTQvOTkgICAgIHN0ZXBoZW4gICAgIFJlbW92ZWQga1Jlc291cmNlQnVuZGxlU3VmZml4CiogMDYvMjIvOTkgICAgIHN0ZXBoZW4gICAgIEZpeGVkIGxvZ2ljIGluIGNvbnN0cnVjdEZyb21GaWxlKCkgc2luY2UgLmN0eAoqICAgICAgICAgICAgICAgICAgICAgICAgICBmaWxlcyBhcmUgbm8gbG9uZ2VyIHVzZWQuCiogMTEvMDIvOTkgICAgIGhlbGVuYSAgICAgIENvbGxhdG9yIHBlcmZvcm1hbmNlIGVuaGFuY2VtZW50cy4gIFNwZWNpYWwgY2FzZQoqICAgICAgICAgICAgICAgICAgICAgICAgICBmb3IgTk9fT1Agc2l0dWF0aW9ucy4KKiAxMS8xNy85OSAgICAgc3JsICAgICAgICAgTW9yZSBwZXJmb3JtYW5jZSBlbmhhbmNlbWVudHMuIElubGluZWQgc29tZSBpbnRlcm5hbCBmdW5jdGlvbnMuCiogMTIvMTUvOTkgICAgIGFsaXUgICAgICAgIFVwZGF0ZSB0byBzdXBwb3J0IFRoYWkgY29sbGF0aW9uLiAgTW92ZSBOb3JtYWxpemVySXRlcmF0b3IKKiAgICAgICAgICAgICAgICAgICAgICAgICAgdG8gaW1wbGVtZW50YXRpb24gZmlsZS4KKiAwMS8yOS8wMSAgICAgc3lud2VlICAgICAgTW9kaWZpZWQgaW50byBhIEMrKyB3cmFwcGVyIGNhbGxpbmcgQyBBUElzICh1Y29sLmgpCiovCgojaW5jbHVkZSAidW5pY29kZS90Ymxjb2xsLmgiCiNpbmNsdWRlICJ1bmljb2RlL2NvbGVpdHIuaCIKI2luY2x1ZGUgInVuaWNvZGUvcmVzYnVuZC5oIgojaW5jbHVkZSAidWNvbF9pbXAuaCIKI2luY2x1ZGUgInVyZXNpbXAuaCIKI2luY2x1ZGUgInVoYXNoLmgiCiNpbmNsdWRlICJjbWVtb3J5LmgiCgovKiBwdWJsaWMgUnVsZUJhc2VkQ29sbGF0b3IgY29uc3RydWN0b3IgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAqLwoKVV9OQU1FU1BBQ0VfQkVHSU4KCi8qKgoqIENvcHkgY29uc3RydWN0b3IKKi8KUnVsZUJhc2VkQ29sbGF0b3I6OlJ1bGVCYXNlZENvbGxhdG9yKGNvbnN0IFJ1bGVCYXNlZENvbGxhdG9yJiB0aGF0KSA6CiAgICAgICAgICAgICAgQ29sbGF0b3IodGhhdCksIGRhdGFJc093bmVkKEZBTFNFKSwgdWNvbGxhdG9yKHRoYXQudWNvbGxhdG9yKSwKICAgICAgICAgICAgICB1cnVsZXN0cmluZyh0aGF0LnVydWxlc3RyaW5nKQp7Cn0KClJ1bGVCYXNlZENvbGxhdG9yOjpSdWxlQmFzZWRDb2xsYXRvcihjb25zdCBVbmljb2RlU3RyaW5nJiBydWxlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgOgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIFVDT0xfREVGQVVMVF9TVFJFTkdUSCwKICAgICAgICAgICAgVUNPTF9ERUZBVUxULAogICAgICAgICAgICBzdGF0dXMpOwp9CgpSdWxlQmFzZWRDb2xsYXRvcjo6UnVsZUJhc2VkQ29sbGF0b3IoY29uc3QgVW5pY29kZVN0cmluZyYgcnVsZXMsCiAgICAgICAgICAgICAgICAgICAgICBFQ29sbGF0aW9uU3RyZW5ndGggY29sbGF0aW9uU3RyZW5ndGgsCiAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIDogZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIGdldFVDb2xsYXRpb25TdHJlbmd0aChjb2xsYXRpb25TdHJlbmd0aCksCiAgICAgICAgICAgIFVDT0xfREVGQVVMVCwKICAgICAgICAgICAgc3RhdHVzKTsKfQoKLy8gVE9ETyB0aGlzIGlzIGEgZGVwcmVjYXRlZCBjb25zdHJ1Y3RvciwgcmVtb3ZlID4yMDAyLXNlcC0zMApSdWxlQmFzZWRDb2xsYXRvcjo6UnVsZUJhc2VkQ29sbGF0b3IoY29uc3QgVW5pY29kZVN0cmluZyYgcnVsZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBOb3JtYWxpemVyOjpFTW9kZSBkZWNvbXBvc2l0aW9uTW9kZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgOgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIFVDT0xfREVGQVVMVF9TVFJFTkdUSCwKICAgICAgICAgICAgKFVDb2xBdHRyaWJ1dGVWYWx1ZSlOb3JtYWxpemVyOjpnZXRVTm9ybWFsaXphdGlvbk1vZGUoZGVjb21wb3NpdGlvbk1vZGUsIHN0YXR1cyksCiAgICAgICAgICAgIHN0YXR1cyk7Cn0KClJ1bGVCYXNlZENvbGxhdG9yOjpSdWxlQmFzZWRDb2xsYXRvcihjb25zdCBVbmljb2RlU3RyaW5nJiBydWxlcywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVDb2xBdHRyaWJ1dGVWYWx1ZSBkZWNvbXBvc2l0aW9uTW9kZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgOgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIFVDT0xfREVGQVVMVF9TVFJFTkdUSCwKICAgICAgICAgICAgZGVjb21wb3NpdGlvbk1vZGUsCiAgICAgICAgICAgIHN0YXR1cyk7Cn0KCi8vIFRPRE8gdGhpcyBpcyBhIGRlcHJlY2F0ZWQgY29uc3RydWN0b3IsIHJlbW92ZSA+MjAwMi1zZXAtMzAKUnVsZUJhc2VkQ29sbGF0b3I6OlJ1bGVCYXNlZENvbGxhdG9yKGNvbnN0IFVuaWNvZGVTdHJpbmcmIHJ1bGVzLAogICAgICAgICAgICAgICAgICAgICAgRUNvbGxhdGlvblN0cmVuZ3RoIGNvbGxhdGlvblN0cmVuZ3RoLAogICAgICAgICAgICAgICAgICAgICAgTm9ybWFsaXplcjo6RU1vZGUgZGVjb21wb3NpdGlvbk1vZGUsCiAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIDogZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIGdldFVDb2xsYXRpb25TdHJlbmd0aChjb2xsYXRpb25TdHJlbmd0aCksCiAgICAgICAgICAgIChVQ29sQXR0cmlidXRlVmFsdWUpTm9ybWFsaXplcjo6Z2V0VU5vcm1hbGl6YXRpb25Nb2RlKGRlY29tcG9zaXRpb25Nb2RlLCBzdGF0dXMpLAogICAgICAgICAgICBzdGF0dXMpOwp9CgpSdWxlQmFzZWRDb2xsYXRvcjo6UnVsZUJhc2VkQ29sbGF0b3IoY29uc3QgVW5pY29kZVN0cmluZyYgcnVsZXMsCiAgICAgICAgICAgICAgICAgICAgICBFQ29sbGF0aW9uU3RyZW5ndGggY29sbGF0aW9uU3RyZW5ndGgsCiAgICAgICAgICAgICAgICAgICAgICBVQ29sQXR0cmlidXRlVmFsdWUgZGVjb21wb3NpdGlvbk1vZGUsCiAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIDogZGF0YUlzT3duZWQoRkFMU0UpCnsKICBjb25zdHJ1Y3QocnVsZXMsCiAgICAgICAgICAgIGdldFVDb2xsYXRpb25TdHJlbmd0aChjb2xsYXRpb25TdHJlbmd0aCksCiAgICAgICAgICAgIGRlY29tcG9zaXRpb25Nb2RlLAogICAgICAgICAgICBzdGF0dXMpOwp9Cgp2b2lkClJ1bGVCYXNlZENvbGxhdG9yOjpjb25zdHJ1Y3QoY29uc3QgVW5pY29kZVN0cmluZyYgcnVsZXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUNvbEF0dHJpYnV0ZVZhbHVlIGNvbGxhdGlvblN0cmVuZ3RoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVDb2xBdHRyaWJ1dGVWYWx1ZSBkZWNvbXBvc2l0aW9uTW9kZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpCnsKICB1Y29sbGF0b3IgPSB1Y29sX29wZW5SdWxlcyhydWxlcy5nZXRCdWZmZXIoKSwgcnVsZXMubGVuZ3RoKCksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVjb21wb3NpdGlvbk1vZGUsIGNvbGxhdGlvblN0cmVuZ3RoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIE5VTEwsICZzdGF0dXMpOwogIGlmIChVX1NVQ0NFU1Moc3RhdHVzKSkKICB7CiAgICBpbnQzMl90IGxlbmd0aDsKICAgIGNvbnN0IFVDaGFyICpyID0gdWNvbF9nZXRSdWxlcyh1Y29sbGF0b3IsICZsZW5ndGgpOwogICAgaWYgKGxlbmd0aCA+IDApIHsKICAgICAgICAvLyBhbGlhcyB0aGUgcnVsZXMgc3RyaW5nCiAgICAgICAgdXJ1bGVzdHJpbmcgPSBuZXcgVW5pY29kZVN0cmluZyhUUlVFLCByLCBsZW5ndGgpOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgdXJ1bGVzdHJpbmcgPSBuZXcgVW5pY29kZVN0cmluZygpOwogICAgfQogICAgLyogdGVzdCBmb3IgTlVMTCAqLwogICAgaWYgKHVydWxlc3RyaW5nID09IDApIHsKICAgICAgICBzdGF0dXMgPSBVX01FTU9SWV9BTExPQ0FUSU9OX0VSUk9SOwogICAgICAgIHJldHVybjsKICAgIH0KICAgIGRhdGFJc093bmVkID0gVFJVRTsKICB9Cn0KCi8qIFJ1bGVCYXNlZENvbGxhdG9yIHB1YmxpYyBkZXN0cnVjdG9yIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tICovCgpSdWxlQmFzZWRDb2xsYXRvcjo6flJ1bGVCYXNlZENvbGxhdG9yKCkKewogIGlmIChkYXRhSXNPd25lZCkKICB7CiAgICB1Y29sX2Nsb3NlKHVjb2xsYXRvcik7CiAgICBkZWxldGUgdXJ1bGVzdHJpbmc7CiAgfQogIHVjb2xsYXRvciA9IDA7CiAgdXJ1bGVzdHJpbmcgPSAwOwp9CgovKiBSdWxlQmFzZUNvbGxhdG9yIHB1YmxpYyBtZXRob2RzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAqLwoKVUJvb2wgUnVsZUJhc2VkQ29sbGF0b3I6Om9wZXJhdG9yPT0oY29uc3QgQ29sbGF0b3ImIHRoYXQpIGNvbnN0CnsKICAvKiBvbmx5IGNoZWNrcyBmb3IgYWRkcmVzcyBlcXVhbHMgaGVyZSAqLwogIGlmIChDb2xsYXRvcjo6b3BlcmF0b3I9PSh0aGF0KSkKICAgIHJldHVybiBUUlVFOwoKICBpZiAoZ2V0RHluYW1pY0NsYXNzSUQoKSAhPSB0aGF0LmdldER5bmFtaWNDbGFzc0lEKCkpCiAgICByZXR1cm4gRkFMU0U7ICAvKiBub3QgdGhlIHNhbWUgY2xhc3MgKi8KCiAgUnVsZUJhc2VkQ29sbGF0b3ImIHRoYXRBbGlhcyA9IChSdWxlQmFzZWRDb2xsYXRvciYpdGhhdDsKCiAgLyoKICBzeW53ZWUgOiBvcmdpbmFsIGNvZGUgZG9lcyBub3QgY2hlY2sgZm9yIGRhdGEgY29tcGF0aWJpbGl0eQogICovCiAgaWYgKHVjb2xsYXRvciAhPSB0aGF0QWxpYXMudWNvbGxhdG9yKQogICAgcmV0dXJuIEZBTFNFOwoKICByZXR1cm4gVFJVRTsKfQoKUnVsZUJhc2VkQ29sbGF0b3ImIFJ1bGVCYXNlZENvbGxhdG9yOjpvcGVyYXRvcj0oY29uc3QgUnVsZUJhc2VkQ29sbGF0b3ImIHRoYXQpCnsKICBpZiAodGhpcyAhPSAmdGhhdCkKICB7CiAgICBpZiAoZGF0YUlzT3duZWQpCiAgICB7CiAgICAgIHVjb2xfY2xvc2UodWNvbGxhdG9yKTsKICAgICAgdWNvbGxhdG9yID0gTlVMTDsKICAgICAgZGVsZXRlIHVydWxlc3RyaW5nOwogICAgfQoKICAgIGRhdGFJc093bmVkID0gRkFMU0U7CiAgICB1Y29sbGF0b3IgPSB0aGF0LnVjb2xsYXRvcjsKICAgIHVydWxlc3RyaW5nID0gdGhhdC51cnVsZXN0cmluZzsKICB9CiAgcmV0dXJuICp0aGlzOwp9CgpDb2xsYXRvciogUnVsZUJhc2VkQ29sbGF0b3I6OmNsb25lKCkgY29uc3QKewogIHJldHVybiBuZXcgUnVsZUJhc2VkQ29sbGF0b3IoKnRoaXMpOwp9CgpDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IqIFJ1bGVCYXNlZENvbGxhdG9yOjpjcmVhdGVDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIChjb25zdCBVbmljb2RlU3RyaW5nJiBzb3VyY2UpIGNvbnN0CnsKICBVRXJyb3JDb2RlIHN0YXR1cyA9IFVfWkVST19FUlJPUjsKICBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IgKnJlc3VsdCA9IG5ldyBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3Ioc291cmNlLCB0aGlzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXMpOwogIGlmIChVX0ZBSUxVUkUoc3RhdHVzKSkgewogICAgZGVsZXRlIHJlc3VsdDsKICAgIHJldHVybiBOVUxMOwogIH0KCiAgcmV0dXJuIHJlc3VsdDsKfQoKLyoqCiogQ3JlYXRlIGEgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yIG9iamVjdCB0aGF0IHdpbGwgaXRlcmF0b3Igb3ZlciB0aGUKKiBlbGVtZW50cyBpbiBhIHN0cmluZywgdXNpbmcgdGhlIGNvbGxhdGlvbiBydWxlcyBkZWZpbmVkIGluIHRoaXMKKiBSdWxlQmFzZWRDb2xsYXRvcgoqLwpDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IqIFJ1bGVCYXNlZENvbGxhdG9yOjpjcmVhdGVDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGNvbnN0IENoYXJhY3Rlckl0ZXJhdG9yJiBzb3VyY2UpIGNvbnN0CnsKICBVRXJyb3JDb2RlIHN0YXR1cyA9IFVfWkVST19FUlJPUjsKICBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IgKnJlc3VsdCA9IG5ldyBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3Ioc291cmNlLCB0aGlzLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXMpOwoKICBpZiAoVV9GQUlMVVJFKHN0YXR1cykpIHsKICAgIGRlbGV0ZSByZXN1bHQ7CiAgICByZXR1cm4gTlVMTDsKICB9CgogIHJldHVybiByZXN1bHQ7Cn0KCi8qKgoqIFJldHVybiBhIHN0cmluZyByZXByZXNlbnRhdGlvbiBvZiB0aGlzIGNvbGxhdG9yJ3MgcnVsZXMuIFRoZSBzdHJpbmcgY2FuCiogbGF0ZXIgYmUgcGFzc2VkIHRvIHRoZSBjb25zdHJ1Y3RvciB0aGF0IHRha2VzIGEgVW5pY29kZVN0cmluZyBhcmd1bWVudCwKKiB3aGljaCB3aWxsIGNvbnN0cnVjdCBhIGNvbGxhdG9yIHRoYXQncyBmdW5jdGlvbmFsbHkgaWRlbnRpY2FsIHRvIHRoaXMgb25lLgoqIFlvdSBjYW4gYWxzbyBhbGxvdyB1c2VycyB0byBlZGl0IHRoZSBzdHJpbmcgaW4gb3JkZXIgdG8gY2hhbmdlIHRoZSBjb2xsYXRpb24KKiBkYXRhLCBvciB5b3UgY2FuIHByaW50IGl0IG91dCBmb3IgaW5zcGVjdGlvbiwgb3Igd2hhdGV2ZXIuCiovCmNvbnN0IFVuaWNvZGVTdHJpbmcmIFJ1bGVCYXNlZENvbGxhdG9yOjpnZXRSdWxlcygpIGNvbnN0CnsKICAgIHJldHVybiAoKnVydWxlc3RyaW5nKTsKfQoKdm9pZCBSdWxlQmFzZWRDb2xsYXRvcjo6Z2V0UnVsZXMoVUNvbFJ1bGVPcHRpb24gZGVsdGEsIFVuaWNvZGVTdHJpbmcgJmJ1ZmZlcikKewogICAgaW50MzJfdCBydWxlc2l6ZSA9IHVjb2xfZ2V0UnVsZXNFeCh1Y29sbGF0b3IsIGRlbHRhLCBOVUxMLCAtMSk7CgogICAgaWYgKHJ1bGVzaXplID4gMCkgewogICAgICAgIFVDaGFyICpydWxlcyA9IChVQ2hhciopIHVwcnZfbWFsbG9jKCBzaXplb2YoVUNoYXIpICogKHJ1bGVzaXplKSApOwoKICAgICAgICB1Y29sX2dldFJ1bGVzRXgodWNvbGxhdG9yLCBkZWx0YSwgcnVsZXMsIHJ1bGVzaXplKTsKICAgICAgICBidWZmZXIuc2V0VG8ocnVsZXMsIHJ1bGVzaXplKTsKICAgICAgICB1cHJ2X2ZyZWUocnVsZXMpOwogICAgfQogICAgZWxzZSB7CiAgICAgICAgYnVmZmVyLnJlbW92ZSgpOwogICAgfQp9Cgp2b2lkIFJ1bGVCYXNlZENvbGxhdG9yOjpnZXRWZXJzaW9uKFVWZXJzaW9uSW5mbyB2ZXJzaW9uSW5mbykgY29uc3QKewogICAgaWYgKHZlcnNpb25JbmZvIT1OVUxMKXsKICAgICAgICB1Y29sX2dldFZlcnNpb24odWNvbGxhdG9yLCB2ZXJzaW9uSW5mbyk7CiAgICB9Cn0KCkNvbGxhdG9yOjpFQ29tcGFyaXNvblJlc3VsdCBSdWxlQmFzZWRDb2xsYXRvcjo6Y29tcGFyZSgKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBVbmljb2RlU3RyaW5nJiBzb3VyY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgVW5pY29kZVN0cmluZyYgdGFyZ2V0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgbGVuZ3RoKSBjb25zdAp7CiAgcmV0dXJuIGNvbXBhcmUoc291cmNlLmdldEJ1ZmZlcigpLCB1cHJ2X21pbihsZW5ndGgsc291cmNlLmxlbmd0aCgpKSwgdGFyZ2V0LmdldEJ1ZmZlcigpLCB1cHJ2X21pbihsZW5ndGgsdGFyZ2V0Lmxlbmd0aCgpKSk7Cn0KCkNvbGxhdG9yOjpFQ29tcGFyaXNvblJlc3VsdCBSdWxlQmFzZWRDb2xsYXRvcjo6Y29tcGFyZShjb25zdCBVQ2hhciogc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBzb3VyY2VMZW5ndGgsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBVQ2hhciogdGFyZ2V0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCB0YXJnZXRMZW5ndGgpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdAp7CiAgcmV0dXJuIGdldEVDb21wYXJpc29uUmVzdWx0KHVjb2xfc3RyY29sbCh1Y29sbGF0b3IsIHNvdXJjZSwgc291cmNlTGVuZ3RoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRhcmdldCwgdGFyZ2V0TGVuZ3RoKSk7Cn0KCi8qKgoqIENvbXBhcmUgdHdvIHN0cmluZ3MgdXNpbmcgdGhpcyBjb2xsYXRvcgoqLwpDb2xsYXRvcjo6RUNvbXBhcmlzb25SZXN1bHQgUnVsZUJhc2VkQ29sbGF0b3I6OmNvbXBhcmUoCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFVuaWNvZGVTdHJpbmcmIHNvdXJjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgVW5pY29kZVN0cmluZyYgdGFyZ2V0KSBjb25zdAp7CiAgcmV0dXJuIGNvbXBhcmUoc291cmNlLmdldEJ1ZmZlcigpLCBzb3VyY2UubGVuZ3RoKCksIHRhcmdldC5nZXRCdWZmZXIoKSwgdGFyZ2V0Lmxlbmd0aCgpKTsKfQoKLyoqCiogUmV0cmlldmUgYSBjb2xsYXRpb24ga2V5IGZvciB0aGUgc3BlY2lmaWVkIHN0cmluZy4gVGhlIGtleSBjYW4gYmUgY29tcGFyZWQKKiB3aXRoIG90aGVyIGNvbGxhdGlvbiBrZXlzIHVzaW5nIGEgYml0d2lzZSBjb21wYXJpc29uIChlLmcuIG1lbWNtcCkgdG8gZmluZAoqIHRoZSBvcmRlcmluZyBvZiB0aGVpciByZXNwZWN0aXZlIHNvdXJjZSBzdHJpbmdzLiBUaGlzIGlzIGhhbmR5IHdoZW4gZG9pbmcgYQoqIHNvcnQsIHdoZXJlIGVhY2ggc29ydCBrZXkgbXVzdCBiZSBjb21wYXJlZCBtYW55IHRpbWVzLgoqCiogVGhlIGJhc2ljIGFsZ29yaXRobSBoZXJlIGlzIHRvIGZpbmQgYWxsIG9mIHRoZSBjb2xsYXRpb24gZWxlbWVudHMgZm9yIGVhY2gKKiBjaGFyYWN0ZXIgaW4gdGhlIHNvdXJjZSBzdHJpbmcsIGNvbnZlcnQgdGhlbSB0byBhbiBBU0NJSSByZXByZXNlbnRhdGlvbiwgYW5kCiogcHV0IHRoZW0gaW50byB0aGUgY29sbGF0aW9uIGtleS4gIEJ1dCBpdCdzIHRyaWNraWVyIHRoYW4gdGhhdC4gRWFjaAoqIGNvbGxhdGlvbiBlbGVtZW50IGluIGEgc3RyaW5nIGhhcyB0aHJlZSBjb21wb25lbnRzOiBwcmltYXJ5ICgnQScgdnMgJ0InKSwKKiBzZWNvbmRhcnkgKCd1JyB2cyAn/CcpLCBhbmQgdGVydGlhcnkgKCdBJyB2cyAnYScpLCBhbmQgYSBwcmltYXJ5IGRpZmZlcmVuY2UKKiBhdCB0aGUgZW5kIG9mIGEgc3RyaW5nIHRha2VzIHByZWNlZGVuY2Ugb3ZlciBhIHNlY29uZGFyeSBvciB0ZXJ0aWFyeQoqIGRpZmZlcmVuY2UgZWFybGllciBpbiB0aGUgc3RyaW5nLgoqCiogVG8gYWNjb3VudCBmb3IgdGhpcywgd2UgcHV0IGFsbCBvZiB0aGUgcHJpbWFyeSBvcmRlcnMgYXQgdGhlIGJlZ2lubmluZyBvZgoqIHRoZSBzdHJpbmcsIGZvbGxvd2VkIGJ5IHRoZSBzZWNvbmRhcnkgYW5kIHRlcnRpYXJ5IG9yZGVycy4gRWFjaCBzZXQgb2YKKiBvcmRlcnMgaXMgdGVybWluYXRlZCBieSBudWxscyBzbyB0aGF0IGEga2V5IGZvciBhIHN0cmluZyB3aGljaCBpcyBhIGluaXRpYWwKKiBzdWJzdHJpbmcgb2YgYW5vdGhlciBrZXkgd2lsbCBjb21wYXJlIGxlc3Mgd2l0aG91dCBhbnkgc3BlY2lhbCBjYXNlLgoqCiogSGVyZSdzIGEgaHlwb3RoZXRpY2FsIGV4YW1wbGUsIHdpdGggdGhlIGNvbGxhdGlvbiBlbGVtZW50IHJlcHJlc2VudGVkIGFzIGEKKiB0aHJlZS1kaWdpdCBudW1iZXIsIG9uZSBkaWdpdCBmb3IgcHJpbWFyeSwgb25lIGZvciBzZWNvbmRhcnksIGV0Yy4KKgoqIFN0cmluZzogICAgICAgICAgICAgIEEgICAgIGEgICAgIEIgICAgyQoqIENvbGxhdGlvbiBFbGVtZW50czogMTAxICAgMTAwICAgMjAxICA1MTEKKiBDb2xsYXRpb24gS2V5OiAgICAgIDExMjU8bnVsbD4wMDAxPG51bGw+MTAxMTxudWxsPgoqCiogVG8gbWFrZSB0aGluZ3MgZXZlbiB0cmlja2llciwgc2Vjb25kYXJ5IGRpZmZlcmVuY2VzIChhY2NlbnQgbWFya3MpIGFyZQoqIGNvbXBhcmVkIHN0YXJ0aW5nIGF0IHRoZSAqZW5kKiBvZiB0aGUgc3RyaW5nIGluIGxhbmd1YWdlcyB3aXRoIEZyZW5jaAoqIHNlY29uZGFyeSBvcmRlcmluZy4gQnV0IHdoZW4gY29tcGFyaW5nIHRoZSBhY2NlbnQgbWFya3Mgb24gYSBzaW5nbGUgYmFzZQoqIGNoYXJhY3RlciwgdGhleSBhcmUgY29tcGFyZWQgZnJvbSB0aGUgYmVnaW5uaW5nLiBUbyBoYW5kbGUgdGhpcywgd2UgcmV2ZXJzZQoqIGFsbCBvZiB0aGUgYWNjZW50cyB0aGF0IGJlbG9uZyB0byBlYWNoIGJhc2UgY2hhcmFjdGVyLCB0aGVuIHdlIHJldmVyc2UgdGhlCiogZW50aXJlIHN0cmluZyBvZiBzZWNvbmRhcnkgb3JkZXJpbmdzIGF0IHRoZSBlbmQuCiovCkNvbGxhdGlvbktleSYgUnVsZUJhc2VkQ29sbGF0b3I6OmdldENvbGxhdGlvbktleSgKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCBVbmljb2RlU3RyaW5nJiBzb3VyY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ29sbGF0aW9uS2V5JiBzb3J0a2V5LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgY29uc3QKewogIHJldHVybiBnZXRDb2xsYXRpb25LZXkoc291cmNlLmdldEJ1ZmZlcigpLCBzb3VyY2UubGVuZ3RoKCksIHNvcnRrZXksIHN0YXR1cyk7Cn0KCkNvbGxhdGlvbktleSYgUnVsZUJhc2VkQ29sbGF0b3I6OmdldENvbGxhdGlvbktleShjb25zdCBVQ2hhciogc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBzb3VyY2VMZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBDb2xsYXRpb25LZXkmIHNvcnRrZXksCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0CnsKICBpZiAoVV9GQUlMVVJFKHN0YXR1cykpCiAgewogICAgcmV0dXJuIHNvcnRrZXkuc2V0VG9Cb2d1cygpOwogIH0KCiAgaWYgKCghc291cmNlKSB8fCAoc291cmNlTGVuID09IDApKSB7CiAgICByZXR1cm4gc29ydGtleS5yZXNldCgpOwogIH0KCiAgdWludDhfdCAqcmVzdWx0OwogIGludDMyX3QgcmVzdWx0TGVuID0gdWNvbF9nZXRTb3J0S2V5V2l0aEFsbG9jYXRpb24odWNvbGxhdG9yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgc291cmNlLCBzb3VyY2VMZW4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJnN0YXR1cyk7CiAgc29ydGtleS5hZG9wdChyZXN1bHQsIHJlc3VsdExlbik7CiAgcmV0dXJuIHNvcnRrZXk7Cn0KCi8qKgogKiBSZXR1cm4gdGhlIG1heGltdW0gbGVuZ3RoIG9mIGFueSBleHBhbnNpb24gc2VxdWVuY2VzIHRoYXQgZW5kIHdpdGggdGhlCiAqIHNwZWNpZmllZCBjb21wYXJpc29uIG9yZGVyLgogKiBAcGFyYW0gb3JkZXIgYSBjb2xsYXRpb24gb3JkZXIgcmV0dXJuZWQgYnkgcHJldmlvdXMgb3IgbmV4dC4KICogQHJldHVybiB0aGUgbWF4aW11bSBsZW5ndGggb2YgYW55IGV4cGFuc2lvbiBzZXVlbmNlcyBlbmRpbmcgd2l0aCB0aGUKICogICAgICAgICBzcGVjaWZpZWQgb3JkZXIgb3IgMSBpZiBjb2xsYXRpb24gb3JkZXIgZG9lcyBub3Qgb2NjdXIgYXQgdGhlIGVuZCBvZiBhbnkKICogICAgICAgICBleHBhbnNpb24gc2VxdWVuY2UuCiAqIEBzZWUgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yI2dldE1heEV4cGFuc2lvbgogKi8KaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6Z2V0TWF4RXhwYW5zaW9uKGludDMyX3Qgb3JkZXIpIGNvbnN0CnsKICB1aW50OF90IHJlc3VsdDsKICBVQ09MX0dFVE1BWEVYUEFOU0lPTih1Y29sbGF0b3IsICh1aW50MzJfdClvcmRlciwgcmVzdWx0KTsKICByZXR1cm4gcmVzdWx0Owp9Cgp1aW50OF90KiBSdWxlQmFzZWRDb2xsYXRvcjo6Y2xvbmVSdWxlRGF0YShpbnQzMl90ICZsZW5ndGgsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICZzdGF0dXMpCnsKICByZXR1cm4gdWNvbF9jbG9uZVJ1bGVEYXRhKHVjb2xsYXRvciwgJmxlbmd0aCwgJnN0YXR1cyk7Cn0KCnZvaWQgUnVsZUJhc2VkQ29sbGF0b3I6OnNldEF0dHJpYnV0ZShVQ29sQXR0cmlidXRlIGF0dHIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVQ29sQXR0cmlidXRlVmFsdWUgdmFsdWUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICZzdGF0dXMpCnsKICBpZiAoVV9GQUlMVVJFKHN0YXR1cykpCiAgICByZXR1cm47CiAgdWNvbF9zZXRBdHRyaWJ1dGUodWNvbGxhdG9yLCBhdHRyLCB2YWx1ZSwgJnN0YXR1cyk7Cn0KClVDb2xBdHRyaWJ1dGVWYWx1ZSBSdWxlQmFzZWRDb2xsYXRvcjo6Z2V0QXR0cmlidXRlKFVDb2xBdHRyaWJ1dGUgYXR0ciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSAmc3RhdHVzKQp7CiAgaWYgKFVfRkFJTFVSRShzdGF0dXMpKQogICAgcmV0dXJuIFVDT0xfREVGQVVMVDsKICByZXR1cm4gdWNvbF9nZXRBdHRyaWJ1dGUodWNvbGxhdG9yLCBhdHRyLCAmc3RhdHVzKTsKfQoKdWludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OnNldFZhcmlhYmxlVG9wKGNvbnN0IFVDaGFyICp2YXJUb3AsIGludDMyX3QgbGVuLCBVRXJyb3JDb2RlICZzdGF0dXMpIHsKICByZXR1cm4gdWNvbF9zZXRWYXJpYWJsZVRvcCh1Y29sbGF0b3IsIHZhclRvcCwgbGVuLCAmc3RhdHVzKTsKfQoKdWludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OnNldFZhcmlhYmxlVG9wKGNvbnN0IFVuaWNvZGVTdHJpbmcgdmFyVG9wLCBVRXJyb3JDb2RlICZzdGF0dXMpIHsKICByZXR1cm4gdWNvbF9zZXRWYXJpYWJsZVRvcCh1Y29sbGF0b3IsIHZhclRvcC5nZXRCdWZmZXIoKSwgdmFyVG9wLmxlbmd0aCgpLCAmc3RhdHVzKTsKfQoKdm9pZCBSdWxlQmFzZWRDb2xsYXRvcjo6c2V0VmFyaWFibGVUb3AoY29uc3QgdWludDMyX3QgdmFyVG9wLCBVRXJyb3JDb2RlICZzdGF0dXMpIHsKICB1Y29sX3Jlc3RvcmVWYXJpYWJsZVRvcCh1Y29sbGF0b3IsIHZhclRvcCwgJnN0YXR1cyk7Cn0KCnVpbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpnZXRWYXJpYWJsZVRvcChVRXJyb3JDb2RlICZzdGF0dXMpIGNvbnN0IHsKICByZXR1cm4gdWNvbF9nZXRWYXJpYWJsZVRvcCh1Y29sbGF0b3IsICZzdGF0dXMpOwp9CgpDb2xsYXRvciogUnVsZUJhc2VkQ29sbGF0b3I6OnNhZmVDbG9uZSh2b2lkKQp7CiAgVUVycm9yQ29kZSBpbnRTdGF0dXMgPSBVX1pFUk9fRVJST1I7CiAgaW50MzJfdCBidWZmZXJzaXplID0gVV9DT0xfU0FGRUNMT05FX0JVRkZFUlNJWkU7CiAgVUNvbGxhdG9yICp1Y29sID0gdWNvbF9zYWZlQ2xvbmUodWNvbGxhdG9yLCBOVUxMLCAmYnVmZmVyc2l6ZSwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJmludFN0YXR1cyk7CiAgaWYgKFVfRkFJTFVSRShpbnRTdGF0dXMpKSB7CiAgICByZXR1cm4gTlVMTDsKICB9CgogIFVuaWNvZGVTdHJpbmcgKnIgPSBuZXcgVW5pY29kZVN0cmluZygqdXJ1bGVzdHJpbmcpOwogIFJ1bGVCYXNlZENvbGxhdG9yICpyZXN1bHQgPSBuZXcgUnVsZUJhc2VkQ29sbGF0b3IodWNvbCwgcik7CiAgcmVzdWx0LT5kYXRhSXNPd25lZCA9IFRSVUU7CiAgcmV0dXJuIHJlc3VsdDsKfQoKCmludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OmdldFNvcnRLZXkoY29uc3QgVW5pY29kZVN0cmluZyYgc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVpbnQ4X3QgKnJlc3VsdCwgaW50MzJfdCByZXN1bHRMZW5ndGgpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QKewogIHJldHVybiB1Y29sX2dldFNvcnRLZXkodWNvbGxhdG9yLCBzb3VyY2UuZ2V0QnVmZmVyKCksIHNvdXJjZS5sZW5ndGgoKSwgcmVzdWx0LCByZXN1bHRMZW5ndGgpOwp9CgppbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpnZXRTb3J0S2V5KGNvbnN0IFVDaGFyICpzb3VyY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBzb3VyY2VMZW5ndGgsIHVpbnQ4X3QgKnJlc3VsdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90IHJlc3VsdExlbmd0aCkgY29uc3QKewogIHJldHVybiB1Y29sX2dldFNvcnRLZXkodWNvbGxhdG9yLCBzb3VyY2UsIHNvdXJjZUxlbmd0aCwgcmVzdWx0LCByZXN1bHRMZW5ndGgpOwp9CgpDb2xsYXRvcjo6RUNvbGxhdGlvblN0cmVuZ3RoIFJ1bGVCYXNlZENvbGxhdG9yOjpnZXRTdHJlbmd0aCh2b2lkKSBjb25zdAp7CiAgVUVycm9yQ29kZSBpbnRTdGF0dXMgPSBVX1pFUk9fRVJST1I7CiAgcmV0dXJuIGdldEVDb2xsYXRpb25TdHJlbmd0aCh1Y29sX2dldEF0dHJpYnV0ZSh1Y29sbGF0b3IsIFVDT0xfU1RSRU5HVEgsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAmaW50U3RhdHVzKSk7Cn0KCnZvaWQgUnVsZUJhc2VkQ29sbGF0b3I6OnNldFN0cmVuZ3RoKEVDb2xsYXRpb25TdHJlbmd0aCBuZXdTdHJlbmd0aCkKewogIFVFcnJvckNvZGUgaW50U3RhdHVzID0gVV9aRVJPX0VSUk9SOwogIFVDb2xsYXRpb25TdHJlbmd0aCBzdHJlbmd0aCA9IGdldFVDb2xsYXRpb25TdHJlbmd0aChuZXdTdHJlbmd0aCk7CiAgdWNvbF9zZXRBdHRyaWJ1dGUodWNvbGxhdG9yLCBVQ09MX1NUUkVOR1RILCBzdHJlbmd0aCwgJmludFN0YXR1cyk7Cn0KCi8qKgoqIENyZWF0ZSBhIGhhc2ggY29kZSBmb3IgdGhpcyBjb2xsYXRpb24uIEp1c3QgaGFzaCB0aGUgbWFpbiBydWxlIHRhYmxlIC0tIHRoYXQKKiBzaG91bGQgYmUgZ29vZCBlbm91Z2ggZm9yIGFsbW9zdCBhbnkgdXNlLgoqLwppbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpoYXNoQ29kZSgpIGNvbnN0CnsKICBpbnQzMl90IGxlbmd0aDsKICBjb25zdCBVQ2hhciAqcnVsZXMgPSB1Y29sX2dldFJ1bGVzKHVjb2xsYXRvciwgJmxlbmd0aCk7CiAgcmV0dXJuIHVoYXNoX2hhc2hVQ2hhcnNOKHJ1bGVzLCBsZW5ndGgpOwp9CgovKioKKiByZXR1cm4gdGhlIGxvY2FsZSBvZiB0aGlzIGNvbGxhdG9yCiovCmNvbnN0IExvY2FsZSBSdWxlQmFzZWRDb2xsYXRvcjo6Z2V0TG9jYWxlKFVMb2NEYXRhTG9jYWxlVHlwZSB0eXBlLCBVRXJyb3JDb2RlICZzdGF0dXMpIGNvbnN0IHsKICBjb25zdCBjaGFyICpyZXN1bHQgPSB1Y29sX2dldExvY2FsZSh1Y29sbGF0b3IsIHR5cGUsICZzdGF0dXMpOwogIGlmKHJlc3VsdCA9PSBOVUxMKSB7CiAgICBMb2NhbGUgcmVzKCIiKTsKICAgIHJlcy5zZXRUb0JvZ3VzKCk7CiAgICByZXR1cm4gcmVzOwogIH0gZWxzZSB7CiAgICByZXR1cm4gTG9jYWxlKHJlc3VsdCk7CiAgfQp9CgovKioKKiBTZXQgdGhlIGRlY29tcG9zaXRpb24gbW9kZSBvZiB0aGUgQ29sbGF0b3Igb2JqZWN0LgoqIEBwYXJhbSB0aGUgbmV3IGRlY29tcG9zaXRpb24gbW9kZQoqIEBzZWUgQ29sbGF0b3IjZ2V0RGVjb21wb3NpdGlvbgoqLwp2b2lkIFJ1bGVCYXNlZENvbGxhdG9yOjpzZXREZWNvbXBvc2l0aW9uKE5vcm1hbGl6ZXI6OkVNb2RlICBtb2RlKQp7CiAgVUVycm9yQ29kZSBzdGF0dXMgPSBVX1pFUk9fRVJST1I7CiAgdWNvbF9zZXROb3JtYWxpemF0aW9uKHVjb2xsYXRvciwgTm9ybWFsaXplcjo6Z2V0VU5vcm1hbGl6YXRpb25Nb2RlKG1vZGUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXR1cykpOwp9CgovKioKKiBHZXQgdGhlIGRlY29tcG9zaXRpb24gbW9kZSBvZiB0aGUgQ29sbGF0b3Igb2JqZWN0LgoqIEByZXR1cm4gdGhlIGRlY29tcG9zaXRpb24gbW9kZQoqIEBzZWUgQ29sbGF0b3Ijc2V0RGVjb21wb3NpdGlvbgoqLwpOb3JtYWxpemVyOjpFTW9kZSBSdWxlQmFzZWRDb2xsYXRvcjo6Z2V0RGVjb21wb3NpdGlvbih2b2lkKSBjb25zdAp7CiAgVUVycm9yQ29kZSBzdGF0dXMgPSBVX1pFUk9fRVJST1I7CiAgcmV0dXJuIE5vcm1hbGl6ZXI6OmdldE5vcm1hbGl6ZXJFTW9kZSh1Y29sX2dldE5vcm1hbGl6YXRpb24odWNvbGxhdG9yKSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0dXMpOwp9CgovLyBSdWxlQmFzZUNvbGxhdG9yTmV3IHByaXZhdGUgY29uc3RydWN0b3IgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKUnVsZUJhc2VkQ29sbGF0b3I6OlJ1bGVCYXNlZENvbGxhdG9yKCkgOiBkYXRhSXNPd25lZChGQUxTRSksIHVjb2xsYXRvcigwKQp7Cn0KClJ1bGVCYXNlZENvbGxhdG9yOjpSdWxlQmFzZWRDb2xsYXRvcihVQ29sbGF0b3IgKmNvbGxhdG9yLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyAqcnVsZSkgOiBkYXRhSXNPd25lZChGQUxTRSkKewogIHVjb2xsYXRvciA9IGNvbGxhdG9yOwogIHVydWxlc3RyaW5nID0gcnVsZTsKfQoKUnVsZUJhc2VkQ29sbGF0b3I6OlJ1bGVCYXNlZENvbGxhdG9yKGNvbnN0IExvY2FsZSYgZGVzaXJlZExvY2FsZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgOgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGF0YUlzT3duZWQoRkFMU0UpLCB1Y29sbGF0b3IoMCkKewogIGlmIChVX0ZBSUxVUkUoc3RhdHVzKSkKICAgIHJldHVybjsKCiAgLyoKICBUcnkgdG8gbG9hZCwgaW4gb3JkZXI6CiAgIDEuIFRoZSBkZXNpcmVkIGxvY2FsZSdzIGNvbGxhdGlvbi4KICAgMi4gQSBmYWxsYmFjayBvZiB0aGUgZGVzaXJlZCBsb2NhbGUuCiAgIDMuIFRoZSBkZWZhdWx0IGxvY2FsZSdzIGNvbGxhdGlvbi4KICAgNC4gQSBmYWxsYmFjayBvZiB0aGUgZGVmYXVsdCBsb2NhbGUuCiAgIDUuIFRoZSBkZWZhdWx0IGNvbGxhdGlvbiBydWxlcywgd2hpY2ggY29udGFpbnMgZW5fVVMgY29sbGF0aW9uIHJ1bGVzLgoKICAgVG8gcmVpdGVyYXRlLCB3ZSB0cnk6CiAgIFNwZWNpZmljOgogICAgbGFuZ3VhZ2UrY291bnRyeSt2YXJpYW50CiAgICBsYW5ndWFnZStjb3VudHJ5CiAgICBsYW5ndWFnZQogICBEZWZhdWx0OgogICAgbGFuZ3VhZ2UrY291bnRyeSt2YXJpYW50CiAgICBsYW5ndWFnZStjb3VudHJ5CiAgICBsYW5ndWFnZQogICBSb290OiAoYWthIERFRkFVTFRSVUxFUykKICAgc3RlcHMgMS01IGFyZSBoYW5kbGVkIGJ5IHJlc291cmNlIGJ1bmRsZSBmYWxsYmFjayBtZWNoYW5pc20uCiAgIGhvd2V2ZXIsIGluIGEgdmVyeSB1bnByb2JhYmxlIHNpdHVhdGlvbiB0aGF0IG5vIHJlc291cmNlIGJ1bmRsZQogICBkYXRhIGV4aXN0cywgc3RlcCA1IGlzIHJlcGVhdGVkIHdpdGggaGFyZGNvZGVkIGRlZmF1bHQgcnVsZXMuCiAgKi8KCiAgc2V0VUNvbGxhdG9yKGRlc2lyZWRMb2NhbGUsIHN0YXR1cyk7CgogIGlmIChVX0ZBSUxVUkUoc3RhdHVzKSkKICB7CiAgICBzdGF0dXMgPSBVX1pFUk9fRVJST1I7CgogICAgc2V0VUNvbGxhdG9yKGtSb290TG9jYWxlTmFtZSwgc3RhdHVzKTsKICAgIGlmIChzdGF0dXMgPT0gVV9aRVJPX0VSUk9SKSB7CiAgICAgICAgc3RhdHVzID0gVV9VU0lOR19ERUZBVUxUX0VSUk9SOwogICAgfQogIH0KCiAgaWYgKFVfU1VDQ0VTUyhzdGF0dXMpKQogIHsKICAgIGludDMyX3QgbGVuZ3RoOwogICAgY29uc3QgVUNoYXIgKnIgPSB1Y29sX2dldFJ1bGVzKHVjb2xsYXRvciwgJmxlbmd0aCk7CiAgICBpZiAobGVuZ3RoID4gMCkgewogICAgICAgIC8vIGFsaWFzIHRoZSBydWxlcyBzdHJpbmcKICAgICAgICB1cnVsZXN0cmluZyA9IG5ldyBVbmljb2RlU3RyaW5nKFRSVUUsIHIsIGxlbmd0aCk7CiAgICB9CiAgICBlbHNlIHsKICAgICAgICB1cnVsZXN0cmluZyA9IG5ldyBVbmljb2RlU3RyaW5nKCk7CiAgICB9CiAgICAvKiB0ZXN0IGZvciBOVUxMICovCiAgICBpZiAodXJ1bGVzdHJpbmcgPT0gMCkgewogICAgICAgIHN0YXR1cyA9IFVfTUVNT1JZX0FMTE9DQVRJT05fRVJST1I7CiAgICAgICAgcmV0dXJuOwogICAgfQogICAgZGF0YUlzT3duZWQgPSBUUlVFOwogIH0KCiAgcmV0dXJuOwp9CgovKiBSdWxlQmFzZWRDb2xsYXRvciBwcml2YXRlIGRhdGEgbWVtYmVycyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSAqLwoKLyoKICogVE9ETzoKICogVGhlc2Ugc2hvdWxkIHByb2JhYmx5IGJlIGVudW1zICg8PTB4ZmZmZikgb3IgI2RlZmluZXMgKD4weGZmZmYpCiAqIGZvciBiZXR0ZXIgcGVyZm9ybWFuY2UuCiAqIEluY2x1ZGUgdWNvbF9pbXAuaCBhbmQgdXNlIGl0cyBjb25zdGFudHMgaWYgcG9zc2libGUuCiAqIE9ubHkgdXNlZCBpbiBjb2xlaXRyLmg/IQogKiBSZW1vdmUgZnJvbSBoZXJlIQogKi8KCi8qIG5lZWQgbG9vayB1cCBpbiAuY29tbWl0KCkgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6Q0hBUklOREVYID0gMHg3MDAwMDAwMDsKLyogRXhwYW5kIGluZGV4IGZvbGxvd3MgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6RVhQQU5EQ0hBUklOREVYID0gMHg3RTAwMDAwMDsKLyogY29udHJhY3QgaW5kZXhlcyBmb2xsb3dzICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OkNPTlRSQUNUQ0hBUklOREVYID0gMHg3RjAwMDAwMDsKLyogdW5tYXBwZWQgY2hhcmFjdGVyIHZhbHVlcyAqLwpjb25zdCBpbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpVTk1BUFBFRCA9IDB4RkZGRkZGRkY7Ci8qIHByaW1hcnkgc3RyZW5ndGggaW5jcmVtZW50ICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlBSSU1BUllPUkRFUklOQ1JFTUVOVCA9IDB4MDAwMTAwMDA7Ci8qIHNlY29uZGFyeSBzdHJlbmd0aCBpbmNyZW1lbnQgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6U0VDT05EQVJZT1JERVJJTkNSRU1FTlQgPSAweDAwMDAwMTAwOwovKiB0ZXJ0aWFyeSBzdHJlbmd0aCBpbmNyZW1lbnQgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6VEVSVElBUllPUkRFUklOQ1JFTUVOVCA9IDB4MDAwMDAwMDE7Ci8qIG1hc2sgb2ZmIGFueXRoaW5nIGJ1dCBwcmltYXJ5IG9yZGVyICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlBSSU1BUllPUkRFUk1BU0sgPSAweGZmZmYwMDAwOwovKiBtYXNrIG9mZiBhbnl0aGluZyBidXQgc2Vjb25kYXJ5IG9yZGVyICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlNFQ09OREFSWU9SREVSTUFTSyA9IDB4MDAwMGZmMDA7Ci8qIG1hc2sgb2ZmIGFueXRoaW5nIGJ1dCB0ZXJ0aWFyeSBvcmRlciAqLwpjb25zdCBpbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpURVJUSUFSWU9SREVSTUFTSyA9IDB4MDAwMDAwZmY7Ci8qIG1hc2sgb2ZmIGlnbm9yYWJsZSBjaGFyIG9yZGVyICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OklHTk9SQUJMRU1BU0sgPSAweDAwMDBmZmZmOwovKiB1c2Ugb25seSB0aGUgcHJpbWFyeSBkaWZmZXJlbmNlICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlBSSU1BUllESUZGRVJFTkNFT05MWSA9IDB4ZmZmZjAwMDA7Ci8qIHVzZSBvbmx5IHRoZSBwcmltYXJ5IGFuZCBzZWNvbmRhcnkgZGlmZmVyZW5jZSAqLwpjb25zdCBpbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpTRUNPTkRBUllESUZGRVJFTkNFT05MWSA9IDB4ZmZmZmZmMDA7Ci8qIHByaW1hcnkgb3JkZXIgc2hpZnQgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6UFJJTUFSWU9SREVSU0hJRlQgPSAxNjsKLyogc2Vjb25kYXJ5IG9yZGVyIHNoaWZ0ICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlNFQ09OREFSWU9SREVSU0hJRlQgPSA4OwovKiBzdGFydGluZyB2YWx1ZSBmb3IgY29sbGF0aW9uIGVsZW1lbnRzICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OkNPTEVMRU1FTlRTVEFSVCA9IDB4MDIwMjAyMDI7Ci8qIHRlc3RpbmcgbWFzayBmb3IgcHJpbWFyeSBsb3cgZWxlbWVudCAqLwpjb25zdCBpbnQzMl90IFJ1bGVCYXNlZENvbGxhdG9yOjpQUklNQVJZTE9XWkVST01BU0sgPSAweDAwRkYwMDAwOwovKiByZXNldGluZyB2YWx1ZSBmb3Igc2Vjb25kYXJpZXMgYW5kIHRlcnRpYXJpZXMgKi8KY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6UkVTRVRTRUNPTkRBUllURVJUSUFSWSA9IDB4MDAwMDAyMDI7Ci8qIHJlc2V0aW5nIHZhbHVlIGZvciB0ZXJ0aWFyaWVzICovCmNvbnN0IGludDMyX3QgUnVsZUJhc2VkQ29sbGF0b3I6OlJFU0VUVEVSVElBUlkgPSAweDAwMDAwMDAyOwoKY29uc3QgaW50MzJfdCBSdWxlQmFzZWRDb2xsYXRvcjo6UFJJTUlHTk9SQUJMRSA9IDB4MDIwMjsKCi8qIHVuaXF1ZSBmaWxlIGlkIGZvciBwYXJpdHkgY2hlY2sgKi8KY29uc3QgaW50MTZfdCBSdWxlQmFzZWRDb2xsYXRvcjo6RklMRUlEID0gMHg1NDQzOwovKiBiaW5hcnkgY29sbGF0aW9uIGZpbGUgZXh0ZW5zaW9uICovCmNvbnN0IGNoYXIgUnVsZUJhc2VkQ29sbGF0b3I6OmtGaWxlbmFtZVN1ZmZpeFtdID0gIi5jb2wiOwovKiBjbGFzcyBpZCA/IFZhbHVlIGlzIGlycmVsZXZhbnQgKi8KY29uc3QgY2hhciAgUnVsZUJhc2VkQ29sbGF0b3I6OmZnQ2xhc3NJRCA9IDA7CgpVX05BTUVTUEFDRV9FTkQK