LyoKKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKKiAgIENvcHlyaWdodCAoQykgMTk5Ny0xOTk5LCBJbnRlcm5hdGlvbmFsIEJ1c2luZXNzIE1hY2hpbmVzCiogICBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLiAgQWxsIFJpZ2h0cyBSZXNlcnZlZC4KKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKKi8KLy89PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09Ci8vCi8vIEZpbGUgY29sZWl0ci5oCi8vCi8vIAovLwovLyBDcmVhdGVkIGJ5OiBIZWxlbmEgU2hpaAovLwovLyBNb2RpZmljYXRpb24gSGlzdG9yeToKLy8KLy8gIERhdGUgICAgICAgICBOYW1lICAgICAgICAgIERlc2NyaXB0aW9uCi8vCi8vICA4LzE4Lzk3ICAgICBoZWxlbmEgICAgICBBZGRlZCBpbnRlcm5hbCBBUEkgZG9jdW1lbnRhdGlvbi4KLy8gMDgvMDMvOTggICAgICAgIGVybSAgICAgICAgICAgIFN5bmNoZWQgd2l0aCAxLjIgdmVyc2lvbiBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IuamF2YQovLyAxMi8xMC85OSAgICAgIGFsaXUgICAgICAgICAgUG9ydGVkIFRoYWkgY29sbGF0aW9uIHN1cHBvcnQgZnJvbSBKYXZhLgovLz09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCiNpZm5kZWYgQ09MRUlUUl9ICiNkZWZpbmUgQ09MRUlUUl9ICgoKI2luY2x1ZGUgInVuaWNvZGUvdW5pc3RyLmgiCiNpbmNsdWRlICJ1bmljb2RlL3RibGNvbGwuaCIKI2luY2x1ZGUgInVuaWNvZGUvY2hhcml0ZXIuaCIKCmNsYXNzIE5vcm1hbGl6ZXI7CmNsYXNzIFZlY3Rvck9mUFRvQ29udHJhY3RFbGVtZW50OwpjbGFzcyBSdWxlQmFzZWRDb2xsYXRvcjsKCi8qKgogKiBUaGUgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yIGNsYXNzIGlzIHVzZWQgYXMgYW4gaXRlcmF0b3IgdG8gd2FsayB0aHJvdWdoCiAqIGVhY2ggY2hhcmFjdGVyIG9mIGFuIGludGVybmF0aW9uYWwgc3RyaW5nLiBVc2UgdGhlIGl0ZXJhdG9yIHRvIHJldHVybiB0aGUKICogb3JkZXJpbmcgcHJpb3JpdHkgb2YgdGhlIHBvc2l0aW9uZWQgY2hhcmFjdGVyLiBUaGUgb3JkZXJpbmcgcHJpb3JpdHkgb2YKICogYSBjaGFyYWN0ZXIsIHdoaWNoIHdlIHJlZmVyIHRvIGFzIGEga2V5LCBkZWZpbmVzIGhvdyBhIGNoYXJhY3RlciBpcwogKiBjb2xsYXRlZCBpbiB0aGUgZ2l2ZW4gY29sbGF0aW9uIG9iamVjdC4KICogRm9yIGV4YW1wbGUsIGNvbnNpZGVyIHRoZSBmb2xsb3dpbmcgaW4gU3BhbmlzaDoKICogPHByZT4KICogLiAgICAgICAiY2EiIC0+IHRoZSBmaXJzdCBrZXkgaXMga2V5KCdjJykgYW5kIHNlY29uZCBrZXkgaXMga2V5KCdhJykuCiAqIC4gICAgICAgImNoYSIgLT4gdGhlIGZpcnN0IGtleSBpcyBrZXkoJ2NoJykgYW5kIHNlY29uZCBrZXkgaXMga2V5KCdhJykuCiAqIDwvcHJlPgogKiBBbmQgaW4gR2VybWFuLAogKiA8cHJlPgogKiAuICAgICAgICLmYiItPiB0aGUgZmlyc3Qga2V5IGlzIGtleSgnYScpLCB0aGUgc2Vjb25kIGtleSBpcyBrZXkoJ2UnKSwgYW5kCiAqIC4gICAgICAgdGhlIHRoaXJkIGtleSBpcyBrZXkoJ2InKS4KICogPC9wcmU+CiAqIFRoZSBrZXkgb2YgYSBjaGFyYWN0ZXIsIGlzIGFuIGludGVnZXIgY29tcG9zZWQgb2YgcHJpbWFyeSBvcmRlcihzaG9ydCksCiAqIHNlY29uZGFyeSBvcmRlcihjaGFyKSwgYW5kIHRlcnRpYXJ5IG9yZGVyKGNoYXIpLiAgSmF2YSBzdHJpY3RseSBkZWZpbmVzCiAqIHRoZSBzaXplIGFuZCBzaWduZWRuZXNzIG9mIGl0cyBwcmltaXRpdmUgZGF0YSB0eXBlcy4gIFRoZXJlZm9yZSwgdGhlIHN0YXRpYwogKiBmdW5jdGlvbnMgcHJpbWFyeU9yZGVyKCksIHNlY29uZGFyeU9yZGVyKCksIGFuZCB0ZXJ0aWFyeU9yZGVyKCkgcmV0dXJuIGludDMyX3QKICogdG8gZW5zdXJlIHRoZSBjb3JyZWN0bmVzcyBvZiB0aGUga2V5IHZhbHVlLgogKiA8cD5FeGFtcGxlIG9mIHRoZSBpdGVyYXRvciB1c2FnZTogKHdpdGhvdXQgZXJyb3IgY2hlY2tpbmcpCiAqIDxwcmU+CiAqIC4gIHZvaWQgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yX0V4YW1wbGUoKQogKiAuICB7CiAqIC4gICAgICBVbmljb2RlU3RyaW5nIHN0ciA9ICJUaGlzIGlzIGEgdGVzdCI7CiAqIC4gICAgICBVRXJyb3JDb2RlIHN1Y2Nlc3MgPSBVX1pFUk9fRVJST1I7CiAqIC4gICAgICBSdWxlQmFzZWRDb2xsYXRvciogcmJjID0KICogLiAgICAgICAgICAoUnVsZUJhc2VkQ29sbGF0b3IqKSBSdWxlQmFzZWRDb2xsYXRvcjo6Y3JlYXRlSW5zdGFuY2Uoc3VjY2Vzcyk7CiAqIC4gICAgICBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IqIGMgPQogKiAuICAgICAgICAgIHJiYy0+Y3JlYXRlQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yKCBzdHIgKTsKICogLiAgICAgIGludDMyX3Qgb3JkZXIgPSBjLT5uZXh0KHN1Y2Nlc3MpOwogKiAuICAgICAgaW50MzJfdCBwcmltYXJ5T3JkZXIgPSBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OnByaW1hcnlPcmRlciggb3JkZXIgKTsKICogLiAgICAgIGRlbGV0ZSBjOwogKiAuICAgICAgZGVsZXRlIHJiYzsKICogLiAgfQogKiA8L3ByZT4KICogPHA+CiAqIENvbGxhdGlvbkVsZW1lbnRJdGVyYXRvcjo6bmV4dCByZXR1cm5zIHRoZSBjb2xsYXRpb24gb3JkZXIgb2YgdGhlIG5leHQKICogY2hhcmFjdGVyIGJhc2VkIG9uIHRoZSBjb21wYXJpc29uIGxldmVsIG9mIHRoZSBjb2xsYXRvci4gIEEgY29sbGF0aW9uIG9yZGVyIAogKiBjb25zaXN0cyBvZiBwcmltYXJ5IG9yZGVyLCBzZWNvbmRhcnkgb3JkZXIgYW5kIHRlcnRpYXJ5IG9yZGVyLiAgVGhlIGRhdGEgCiAqIHR5cGUgb2YgdGhlIGNvbGxhdGlvbiBvcmRlciBpcyA8c3Ryb25nPmludDMyX3Q8L3N0cm9uZz4uICBUaGUgZmlyc3QgMTYgYml0cyBvZiAKICogYSBjb2xsYXRpb24gb3JkZXIgaXMgaXRzIHByaW1hcnkgb3JkZXI7IHRoZSBuZXh0IDggYml0cyBpcyB0aGUgc2Vjb25kYXJ5IAogKiBvcmRlciBhbmQgdGhlIGxhc3QgOCBiaXRzIGlzIHRoZSB0ZXJ0aWFyeSBvcmRlci4KICoKICogQHNlZSAgICAgICAgICAgICAgICBDb2xsYXRvcgogKiBAc2VlICAgICAgICAgICAgICAgIFJ1bGVCYXNlZENvbGxhdG9yCiAqIEB2ZXJzaW9uICAgICAgICAgICAgMS45IDEvMzAvOTcKICogQGF1dGhvciAgICAgICAgICAgICBIZWxlbmEgU2hpaAogKi8KY2xhc3MgVV9JMThOX0FQSSBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IKewpwdWJsaWM6ICAgICAKICAgIC8qKgogICAgICogTlVMTE9SREVSIGluZGljYXRlcyB0aGUgaXRlcmF0b3IgaGFzIGNvbnN1bWVkIHRoZSBsYXN0IGVsZW1lbnQuCiAgICAgKi8KICAgIHN0YXRpYyAgaW50MzJfdCBjb25zdCAgIE5VTExPUkRFUjsKCiAgICAvKiogRGVzdHJ1Y3RvcgogICAgICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfkNvbGxhdGlvbkVsZW1lbnRJdGVyYXRvcigpOwogICAgLyoqCiAgICAgKiBSZXR1cm5zIHRydWUgaWYgIm90aGVyIiBpcyB0aGUgc2FtZSBhcyAidGhpcyIKICAgICAqLwogICAgdmlydHVhbCAgICAgYm9vbF90ICAgICAgICAgIG9wZXJhdG9yPT0oY29uc3QgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yJiBvdGhlcikgY29uc3Q7CgogICAgLyoqCiAgICAgKiBSZXR1cm5zIHRydWUgaWYgIm90aGVyIiBpcyBub3QgdGhlIHNhbWUgYXMgInRoaXMiLgogICAgICovCiAgICB2aXJ0dWFsICAgICBib29sX3QgICAgICAgICAgb3BlcmF0b3IhPShjb25zdCBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3ImIG90aGVyKSBjb25zdDsKCiAgICAvKioKICAgICAqIFJlc2V0cyB0aGUgY3Vyc29yIHRvIHRoZSBiZWdpbm5pbmcgb2YgdGhlIHN0cmluZy4KICAgICAqLwogICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHJlc2V0KHZvaWQpOwogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBvcmRlcmluZyBwcmlvcml0eSBvZiB0aGUgbmV4dCBjaGFyYWN0ZXIgaW4gdGhlIHN0cmluZy4KICAgICAqIEBwYXJhbSBzdGF0dXMgdGhlIGVycm9yIGNvZGUgc3RhdHVzLgogICAgICogQHJldHVybiB0aGUgbmV4dCBjaGFyYWN0ZXIncyBvcmRlcmluZy4gIFJldHVybnMgTlVMTE9SREVSIGlmCiAgICAgKiB0aGUgZW5kIG9mIHN0cmluZyBpcyByZWFjaGVkLgogICAgICovCiAgICAgICAgICAgIGludDMyX3QgICAgICAgICAgICAgbmV4dChVRXJyb3JDb2RlJiBzdGF0dXMpOwogICAgIC8qKgogICAgICAqIEdldCB0aGUgb3JkZXJpbmcgcHJpb3JpdHkgb2YgdGhlIHByZXZpb3VzIGNvbGxhdGlvbiBlbGVtZW50IGluIHRoZSBzdHJpbmcuCiAgICAgICogQHBhcmFtIHN0YXR1cyB0aGUgZXJyb3IgY29kZSBzdGF0dXMuCiAgICAgICogQHJldHVybiB0aGUgcHJldmlvdXMgZWxlbWVudCdzIG9yZGVyaW5nLiAgUmV0dXJucyBOVUxMT1JERVIgaWYKICAgICAgKiB0aGUgYmVnaW5uaW5nIG9mIHN0cmluZyBpcyByZWFjaGVkLgogICAgICAqLwogICAgICAgICAgICBpbnQzMl90ICAgICAgICAgICAgICAgICBwcmV2aW91cyhVRXJyb3JDb2RlJiBzdGF0dXMpOwoKICAgIC8qKgogICAgICogR2V0cyB0aGUgcHJpbWFyeSBvcmRlciBvZiBhIGNvbGxhdGlvbiBvcmRlci4KICAgICAqIEBwYXJhbSBvcmRlciB0aGUgY29sbGF0aW9uIG9yZGVyCiAgICAgKiBAcmV0dXJuIHRoZSBwcmltYXJ5IG9yZGVyIG9mIGEgY29sbGF0aW9uIG9yZGVyLgogICAgICovCiAgICBzdGF0aWMgIGludDMyX3QgICAgICAgICAgICAgcHJpbWFyeU9yZGVyKGludDMyX3Qgb3JkZXIpOwogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBzZWNvbmRhcnkgb3JkZXIgb2YgYSBjb2xsYXRpb24gb3JkZXIuCiAgICAgKiBAcGFyYW0gb3JkZXIgdGhlIGNvbGxhdGlvbiBvcmRlcgogICAgICogQHJldHVybiB0aGUgc2Vjb25kYXJ5IG9yZGVyIG9mIGEgY29sbGF0aW9uIG9yZGVyLgogICAgICovCiAgICBzdGF0aWMgIGludDMyX3QgICAgICAgICAgICAgc2Vjb25kYXJ5T3JkZXIoaW50MzJfdCBvcmRlcik7CiAgICAvKioKICAgICAqIEdldHMgdGhlIHRlcnRpYXJ5IG9yZGVyIG9mIGEgY29sbGF0aW9uIG9yZGVyLgogICAgICogQHBhcmFtIG9yZGVyIHRoZSBjb2xsYXRpb24gb3JkZXIKICAgICAqIEByZXR1cm4gdGhlIHRlcnRpYXJ5IG9yZGVyIG9mIGEgY29sbGF0aW9uIG9yZGVyLgogICAgICovCiAgICBzdGF0aWMgIGludDMyX3QgICAgICAgICAgICAgdGVydGlhcnlPcmRlcihpbnQzMl90IG9yZGVyKTsKICAgIC8qKgogICAgICogIFJldHVybiB0aGUgbWF4aW11bSBsZW5ndGggb2YgYW55IGV4cGFuc2lvbiBzZXF1ZW5jZXMgdGhhdCBlbmQKICAgICAqICB3aXRoIHRoZSBzcGVjaWZpZWQgY29tcGFyaXNvbiBvcmRlci4KICAgICAqICBAcGFyYW0gb3JkZXIgYSBjb2xsYXRpb24gb3JkZXIgcmV0dXJuZWQgYnkgcHJldmlvdXMgb3IgbmV4dC4KICAgICAqICBAcmV0dXJuIHRoZSBtYXhpbXVtIGxlbmd0aCBvZiBhbnkgZXhwYW5zaW9uIHNlcXVlbmNlcyBlbmRpbmcKICAgICAqICAgICAgICAgIHdpdGggdGhlIHNwZWNpZmllZCBvcmRlci4KICAgICAqLwogICAgICAgIGludDMyX3QgICAgICAgICAgICAgICAgICAgIGdldE1heEV4cGFuc2lvbihpbnQzMl90IG9yZGVyKSBjb25zdDsKCnB1YmxpYzoKICAgIC8qKgogICAgICogIEdldHMgdGhlIGNvbXBhcmlzb24gb3JkZXIgaW4gdGhlIGRlc2lyZWQgc3RyZW5ndGguICBJZ25vcmUgdGhlIG90aGVyCiAgICAgKiAgZGlmZmVyZW5jZXMuCiAgICAgKiAgQHBhcmFtIG9yZGVyIFRoZSBvcmRlciB2YWx1ZQogICAgICovCiAgICAgICAgICAgaW50MzJfdCAgICAgICAgICAgICAgc3RyZW5ndGhPcmRlcihpbnQzMl90IG9yZGVyKSBjb25zdDsKICAgIC8qKgogICAgICogU2V0cyB0aGUgc291cmNlIHN0cmluZy4KICAgICAqIEBwYXJhbSBzdHIgdGhlIHNvdXJjZSBzdHJpbmcuCiAgICAgKiBAcGFyYW0gc3RhdHVzIHRoZSBlcnJvciBjb2RlIHN0YXR1cy4KICAgICAqLwogICAgICAgICAgICB2b2lkICAgICAgICAgICAgICAgIHNldFRleHQoY29uc3QgICBVbmljb2RlU3RyaW5nJiAgc3RyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiAgICAgIHN0YXR1cyk7CiAgICAgLyoqCiAgICAgKiBTZXRzIHRoZSBzb3VyY2Ugc3RyaW5nLgogICAgICogQHBhcmFtIHN0ciB0aGUgc291cmNlIGNoYXJhY3RlciBpdGVyYXRvci4KICAgICAqIEBwYXJhbSBzdGF0dXMgdGhlIGVycm9yIGNvZGUgc3RhdHVzLgogICAgICovCiAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgc2V0VGV4dChDaGFyYWN0ZXJJdGVyYXRvciYgIHN0ciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmICAgICAgICAgICAgc3RhdHVzKTsKICAgLyoqCiAgICAgKiAgQ2hlY2tzIGlmIGEgY29tcGFyaXNvbiBvcmRlciBpcyBpZ25vcmFibGUuCiAgICAgKiAgQHBhcmFtIG9yZGVyIHRoZSBjb2xsYXRpb24gb3JkZXIuCiAgICAgKiAgQHJldHVybiBUUlVFIGlmIGEgY2hhcmFjdGVyIGlzIGlnbm9yYWJsZSwgRkFMU0Ugb3RoZXJ3aXNlLgogICAgICovCiAgICBzdGF0aWMgICAgYm9vbF90ICAgICAgICAgICAgICBpc0lnbm9yYWJsZShpbnQzMl90IG9yZGVyKTsKICAgIC8qKgogICAgICogIEdldHMgdGhlIG9mZnNldCBvZiB0aGUgY3VycmVudGx5IHByb2Nlc3NlZCBjaGFyYWN0ZXIgaW4gdGhlIHNvdXJjZSBzdHJpbmcuCiAgICAgKiAgQHJldHVybiB0aGUgb2Zmc2V0IG9mIHRoZSBjaGFyYWN0ZXIuCiAgICAgKi8KICAgICAgICAgICAgVVRleHRPZmZzZXQgICAgICAgICAgZ2V0T2Zmc2V0KHZvaWQpIGNvbnN0OwogICAgLyoqCiAgICAgKiAgU2V0cyB0aGUgb2Zmc2V0IG9mIHRoZSBjdXJyZW50bHkgcHJvY2Vzc2VkIGNoYXJhY3RlciBpbiB0aGUgc291cmNlIHN0cmluZy4KICAgICAqICBAcGFyYW0gbmV3T2Zmc2V0IHRoZSBuZXcgb2Zmc2V0LgogICAgICogIEBwYXJhbSBzdGF0dXMgdGhlIGVycm9yIGNvZGUgc3RhdHVzLgogICAgICogIEByZXR1cm4gdGhlIG9mZnNldCBvZiB0aGUgY2hhcmFjdGVyLgogICAgICovCiAgICAgICAgICAgIHZvaWQgICAgICAgICAgICAgICAgc2V0T2Zmc2V0KFVUZXh0T2Zmc2V0IG5ld09mZnNldCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKTsKcHJvdGVjdGVkOgogICAgLyoqCiAgICAgKiBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IgY29uc3RydWN0b3IuICBUaGlzIHRha2VzIHRoZSBzb3VyY2Ugc3RyaW5nIGFuZAogICAgICogdGhlIGNvbGxhdGlvbiBvYmplY3QuICBUaGUgY3Vyc29yIHdpbGwgd2FsayB0aHJ1IHRoZSBzb3VyY2Ugc3RyaW5nIGJhc2VkCiAgICAgKiBvbiB0aGUgcHJlZGVmaW5lZCBjb2xsYXRpb24gcnVsZXMuICBJZiB0aGUgc291cmNlIHN0cmluZyBpcyBlbXB0eSwKICAgICAqIE5VTExPUkRFUiB3aWxsIGJlIHJldHVybmVkIG9uIHRoZSBjYWxscyB0byBuZXh0KCkuCiAgICAgKiBAcGFyYW0gc291cmNlVGV4dCB0aGUgc291cmNlIHN0cmluZy4KICAgICAqIEBwYXJhbSBzdGFydE9mZnNldCB0aGUgYmVnaW5uaW5nIG9mZnNldCBvZiB0aGUgc3RyaW5nIHdoZXJlIHRoZSBjdXJzb3IKICAgICAqIHN0YXJ0cyB0aGUgaXRlcmF0aW5nLgogICAgICogQHBhcmFtIGVuZE9mZnNldCB0aGUgZW5kaW5nIG9mZnNldCBvZiB0aGUgc3RyaW5nIHdoZXJlIHRoZSBjdXJzb3IKICAgICAqIHN0b3BzIHRoZSBpdGVyYXRpbmcuCiAgICAgKiBAcGFyYW0gb3JkZXIgdGhlIGNvbGxhdGlvbiBvYmplY3QuCiAgICAgKi8KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3IoICAgY29uc3QgICBVbmljb2RlU3RyaW5nJiAgc291cmNlVGV4dCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgICBSdWxlQmFzZWRDb2xsYXRvciogIG9yZGVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpOwoKICAgIC8qKgogICAgICogQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yIGNvbnN0cnVjdG9yLiAgVGhpcyB0YWtlcyB0aGUgc291cmNlIHN0cmluZyBhbmQKICAgICAqIHRoZSBjb2xsYXRpb24gb2JqZWN0LiAgVGhlIGN1cnNvciB3aWxsIHdhbGsgdGhydSB0aGUgc291cmNlIHN0cmluZyBiYXNlZAogICAgICogb24gdGhlIHByZWRlZmluZWQgY29sbGF0aW9uIHJ1bGVzLiAgSWYgdGhlIHNvdXJjZSBzdHJpbmcgaXMgZW1wdHksCiAgICAgKiBOVUxMT1JERVIgd2lsbCBiZSByZXR1cm5lZCBvbiB0aGUgY2FsbHMgdG8gbmV4dCgpLgogICAgICogQHBhcmFtIHNvdXJjZVRleHQgdGhlIHNvdXJjZSBzdHJpbmcuCiAgICAgKiBAcGFyYW0gc3RhcnRPZmZzZXQgdGhlIGJlZ2lubmluZyBvZmZzZXQgb2YgdGhlIHN0cmluZyB3aGVyZSB0aGUgY3Vyc29yCiAgICAgKiBzdGFydHMgdGhlIGl0ZXJhdGluZy4KICAgICAqIEBwYXJhbSBlbmRPZmZzZXQgdGhlIGVuZGluZyBvZmZzZXQgb2YgdGhlIHN0cmluZyB3aGVyZSB0aGUgY3Vyc29yCiAgICAgKiBzdG9wcyB0aGUgaXRlcmF0aW5nLgogICAgICogQHBhcmFtIG9yZGVyIHRoZSBjb2xsYXRpb24gb2JqZWN0LgogICAgICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yKCAgIGNvbnN0ICAgIENoYXJhY3Rlckl0ZXJhdG9yJiAgc291cmNlVGV4dCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgICBSdWxlQmFzZWRDb2xsYXRvciogIG9yZGVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpOwogICAgLyoqCiAgICAgKiBBc3NpZ25tZW50IG9wZXJhdG9yCiAgICAgKi8KICAgIGNvbnN0ICAgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yJgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9wZXJhdG9yPShjb25zdCBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3ImIG90aGVyKTsKcHVibGljOgogICAgLyoqCiAgICAgKiBDb3B5IGNvbnN0cnVjdG9yLgogICAgICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yKGNvbnN0ICBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3ImIG90aGVyKTsKICAgIC8vPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CiAgICAvLyBwcml2YXRlcwogICAgLy89PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KcHJpdmF0ZToKICAgIC8qKgogICAgICogRGVmYXVsdCBjb25zdHJ1Y3Rvci4KICAgICAqLwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIENvbGxhdGlvbkVsZW1lbnRJdGVyYXRvcigpOwogICAgLyoqCiAgICAgKiBDb3B5IGNvbnN0cnVjdG9yLgogICAgICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgQ29sbGF0aW9uRWxlbWVudEl0ZXJhdG9yKGNvbnN0IFJ1bGVCYXNlZENvbGxhdG9yKiBvcmRlcik7CgogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBvcmRlcmluZyBwcmlvcml0eSBvZiB0aGUgbmV4dCBjb250cmFjdGluZyBjaGFyYWN0ZXIgaW4gdGhlCiAgICAgKiBzdHJpbmcuCiAgICAgKiBAcGFyYW0gY2ggdGhlIHN0YXJ0aW5nIGNoYXJhY3RlciBvZiBhIGNvbnRyYWN0aW5nIGNoYXJhY3RlciB0b2tlbgogICAgICogQHBhcmFtIHN0YXR1cyB0aGUgZXJyb3IgY29kZSBzdGF0dXMuCiAgICAgKiBAcmV0dXJuIHRoZSBuZXh0IGNvbnRyYWN0aW5nIGNoYXJhY3RlcidzIG9yZGVyaW5nLiAgUmV0dXJucyBOVUxMT1JERVIKICAgICAqIGlmIHRoZSBlbmQgb2Ygc3RyaW5nIGlzIHJlYWNoZWQuCiAgICAgKi8KICAgICAgICAgICAgaW50MzJfdCAgICAgICAgICAgICBuZXh0Q29udHJhY3RDaGFyKCAgIFVDaGFyICAgICBjaCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmICBzdGF0dXMpOwoKICAgIC8qKgogICAgICogR2V0cyB0aGUgb3JkZXJpbmcgcHJpb3JpdHkgb2YgdGhlIHByZXZpb3VzIGNvbnRyYWN0aW5nIGNoYXJhY3RlciBpbiB0aGUKICAgICAqIHN0cmluZy4KICAgICAqIEBwYXJhbSBjaCB0aGUgc3RhcnRpbmcgY2hhcmFjdGVyIG9mIGEgY29udHJhY3RpbmcgY2hhcmFjdGVyIHRva2VuCiAgICAgKiBAcGFyYW0gc3RhdHVzIHRoZSBlcnJvciBjb2RlIHN0YXR1cy4KICAgICAqIEByZXR1cm4gdGhlIHByZXZpb3VzIGNvbnRyYWN0aW5nIGNoYXJhY3RlcidzIG9yZGVyaW5nLiAgUmV0dXJucyBOVUxMT1JERVIKICAgICAqIGlmIHRoZSBzdGFydCBvZiBzdHJpbmcgaXMgcmVhY2hlZC4KICAgICAqLwogICAgICAgICAgICBpbnQzMl90ICAgICAgICAgICAgIHByZXZDb250cmFjdENoYXIoICAgVUNoYXIgICAgIGNoLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgIHN0YXR1cyk7CiAgICAKICAgIGlubGluZSBzdGF0aWMgYm9vbF90IGlzVGhhaVByZVZvd2VsKFVDaGFyIGNoKTsKICAgICAgICAgICAgICAgICAKICAgIGlubGluZSBzdGF0aWMgYm9vbF90IGlzVGhhaUJhc2VDb25zb25hbnQoVUNoYXIgY2gpOwogICAgICAgICAgICAgICAgIAogICAgVmVjdG9yT2ZJbnQqIG1ha2VSZW9yZGVyZWRCdWZmZXIoVUNoYXIgY29sRmlyc3QsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90IGxhc3RWYWx1ZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFZlY3Rvck9mSW50KiBsYXN0RXhwYW5zaW9uLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYm9vbF90IGZvcndhcmQsIFVFcnJvckNvZGUmIHN0YXR1cyk7CgogICAgZnJpZW5kICBjbGFzcyAgIFJ1bGVCYXNlZENvbGxhdG9yOwogICAgc3RhdGljICBjb25zdCAgIGludDMyX3QgICAgICAgICBVTk1BUFBFRENIQVJWQUxVRTsKCiAgICAgICAgICAgIE5vcm1hbGl6ZXIqICAgICAgICAgICAgdGV4dDsgICAgICAgLy8gb3duaW5nIAoKICAgICAgICAgICAgVmVjdG9yT2ZJbnQqICAgICAgICBidWZmZXJBbGlhczsgLy8gbm90IG93bmVkCgogICAgLyoqCiAgICAgKiBvd25CdWZmZXIgd2FudHMgdG8gYmUgYSBzdWJvYmplY3QsIG5vdCBhIHBvaW50ZXIsIGJ1dCB0aGF0CiAgICAgKiBtZWFucyBleHBvc2luZyB0aGUgaW50ZXJuYWwgY2xhc3MgVmVjdG9yT2ZJbnQgYnkgI2luY2x1ZGluZyB0aGUKICAgICAqIGludGVybmFsIGhlYWRlciAidGFibGVzLmgiIC0tIG5vdCBhbGxvd2VkISAgb3duQnVmZmVyIGlzIGEKICAgICAqIGZpeGVkLXNpemUgMi1lbGVtZW50IHZlY3RvciB0aGF0IGlzIHVzZWQgdG8gaGFuZGxlIFRoYWkKICAgICAqIGNvbGxhdGlvbjsgYnVmZmVyQWxpYXMgcG9pbnRzIHRvIG93bkJ1ZmZlciBpbiBzb21lIHNpdHVhdGlvbnMuCiAgICAgKiBbajE1OSAtIGFsaXVdCiAgICAgKi8KICAgICAgICAgICAgVmVjdG9yT2ZJbnQqICAgICAgICBvd25CdWZmZXI7CgogICAgLyoqCiAgICAgKiByZW9yZGVyQnVmZmVyIGlzIGNyZWF0ZWQgb24gZGVtYW5kLCBzbyBpdCBkb2Vzbid0IHdhbnQgdG8gYmUKICAgICAqIGEgc3Vib2JqZWN0IC0tIHBvaW50ZXIgaXMgZmluZS4gIEl0IGlzIGNyZWF0ZWQgYW5kIGJ1ZmZlckFsaWFzCiAgICAgKiBpcyBzZXQgdG8gaXQgdW5kZXIgY2VydGFpbiBjb25kaXRpb25zLiAgT25jZSBjcmVhdGVkLCBpdCBpcwogICAgICogcmV1c2VkIGZvciB0aGUgbGlmZSBvZiB0aGlzIG9iamVjdC4gIEJlY2F1c2Ugb2YgdGhlIGltcGxlbWVudGF0aW9uCiAgICAgKiBvZiBWZWN0b3JPZkludCwgaXQgZ3Jvd3MgbW9ub3RvbmljYWxseS4gIFtqMTU5IC0gYWxpdV0KICAgICAqLwogICAgICAgICAgICBWZWN0b3JPZkludCogICAgICAgIHJlb3JkZXJCdWZmZXI7CgogICAgICAgICAgICBpbnQzMl90ICAgICAgICAgICAgIGV4cEluZGV4OwogICAgICAgICAgICBVbmljb2RlU3RyaW5nICAgICAgIGtleTsKICAgIGNvbnN0ICAgUnVsZUJhc2VkQ29sbGF0b3IqICBvcmRlckFsaWFzOwp9OwoKCi8qKgogKiBHZXQgdGhlIHByaW1hcnkgb3JkZXIgb2YgYSBjb2xsYXRpb24gb3JkZXIuCiAqIEBwYXJhbSBvcmRlciB0aGUgY29sbGF0aW9uIG9yZGVyCiAqIEByZXR1cm4gdGhlIHByaW1hcnkgb3JkZXIgb2YgYSBjb2xsYXRpb24gb3JkZXIuCiAqLwppbmxpbmUgaW50MzJfdApDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OnByaW1hcnlPcmRlcihpbnQzMl90IG9yZGVyKQp7CiAgICBvcmRlciAmPSBSdWxlQmFzZWRDb2xsYXRvcjo6UFJJTUFSWU9SREVSTUFTSzsKICAgIHJldHVybiAob3JkZXIgPj4gUnVsZUJhc2VkQ29sbGF0b3I6OlBSSU1BUllPUkRFUlNISUZUKTsKfQovKioKICogR2V0IHRoZSBzZWNvbmRhcnkgb3JkZXIgb2YgYSBjb2xsYXRpb24gb3JkZXIuCiAqIEBwYXJhbSBvcmRlciB0aGUgY29sbGF0aW9uIG9yZGVyCiAqIEByZXR1cm4gdGhlIHNlY29uZGFyeSBvcmRlciBvZiBhIGNvbGxhdGlvbiBvcmRlci4KICovCmlubGluZSBpbnQzMl90CkNvbGxhdGlvbkVsZW1lbnRJdGVyYXRvcjo6c2Vjb25kYXJ5T3JkZXIoaW50MzJfdCBvcmRlcikKewogICAgb3JkZXIgPSBvcmRlciAmIFJ1bGVCYXNlZENvbGxhdG9yOjpTRUNPTkRBUllPUkRFUk1BU0s7CiAgICByZXR1cm4gKG9yZGVyID4+IFJ1bGVCYXNlZENvbGxhdG9yOjpTRUNPTkRBUllPUkRFUlNISUZUKTsKfQovKioKICogR2V0IHRoZSB0ZXJ0aWFyeSBvcmRlciBvZiBhIGNvbGxhdGlvbiBvcmRlci4KICogQHBhcmFtIG9yZGVyIHRoZSBjb2xsYXRpb24gb3JkZXIKICogQHJldHVybiB0aGUgdGVydGlhcnkgb3JkZXIgb2YgYSBjb2xsYXRpb24gb3JkZXIuCiAqLwppbmxpbmUgaW50MzJfdApDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OnRlcnRpYXJ5T3JkZXIoaW50MzJfdCBvcmRlcikKewogICAgcmV0dXJuIChvcmRlciAmPSBSdWxlQmFzZWRDb2xsYXRvcjo6VEVSVElBUllPUkRFUk1BU0spOwp9CgppbmxpbmUgaW50MzJfdApDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OmdldE1heEV4cGFuc2lvbihpbnQzMl90IG9yZGVyKSBjb25zdAp7CiAgICByZXR1cm4gb3JkZXJBbGlhcy0+Z2V0TWF4RXhwYW5zaW9uKG9yZGVyKTsKfQoKaW5saW5lIGJvb2xfdApDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OmlzSWdub3JhYmxlKGludDMyX3Qgb3JkZXIpCnsKICAgIHJldHVybiAocHJpbWFyeU9yZGVyKG9yZGVyKSA9PSAwKTsKfQoKLyoqCiAqIERldGVybWluZSBpZiBhIGNoYXJhY3RlciBpcyBhIFRoYWkgdm93ZWwgKHdoaWNoIHNvcnRzIGFmdGVyCiAqIGl0cyBiYXNlIGNvbnNvbmFudCkuCiAqLwppbmxpbmUgYm9vbF90IENvbGxhdGlvbkVsZW1lbnRJdGVyYXRvcjo6aXNUaGFpUHJlVm93ZWwoVUNoYXIgY2gpIHsKICAgIHJldHVybiAoY2ggPj0gKFVDaGFyKTB4MEU0MCkgJiYgKGNoIDw9IChVQ2hhcikwWDBFNDQpOwp9CgovKioKICogRGV0ZXJtaW5lIGlmIGEgY2hhcmFjdGVyIGlzIGEgVGhhaSBiYXNlIGNvbnNvbmFudAogKi8KaW5saW5lIGJvb2xfdCBDb2xsYXRpb25FbGVtZW50SXRlcmF0b3I6OmlzVGhhaUJhc2VDb25zb25hbnQoVUNoYXIgY2gpIHsKICAgIHJldHVybiAoY2ggPj0gKFVDaGFyKTB4MEUwMSkgJiYgKGNoIDw9IChVQ2hhcikweDBFMkUpOwp9CgojZW5kaWYK