LyoKKiBDb3B5cmlnaHQgqSB7MTk5Ny0xOTk5fSwgSW50ZXJuYXRpb25hbCBCdXNpbmVzcyBNYWNoaW5lcyBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLiBBbGwgUmlnaHRzIFJlc2VydmVkLgoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgoqCiogRmlsZSBOVU1GTVQuSAoqCiogTW9kaWZpY2F0aW9uIEhpc3Rvcnk6CioKKiAgIERhdGUgICAgICAgIE5hbWUgICAgICAgIERlc2NyaXB0aW9uCiogICAwMi8xOS85NyAgICBhbGl1ICAgICAgICBDb252ZXJ0ZWQgZnJvbSBqYXZhLgoqICAgMDMvMTgvOTcgICAgY2xodWFuZyAgICAgVXBkYXRlZCBwZXIgQysrIGltcGxlbWVudGF0aW9uLgoqICAgMDQvMTcvOTcgICAgYWxpdSAgICAgICAgQ2hhbmdlZCBEaWdpdENvdW50IHRvIGludCBwZXIgY29kZSByZXZpZXcuCiogICAgMDcvMjAvOTggICAgc3RlcGhlbiAgICAgICAgSkRLIDEuMiBzeW5jIHVwLiBBZGRlZCBzY2llbnRpZmljIHN1cHBvcnQuCiogICAgICAgICAgICAgICAgICAgICAgICAgICAgQ2hhbmdlZCBuYW1pbmcgY29udmVudGlvbnMgdG8gbWF0Y2ggQysrIGd1aWRlbGluZXMKKiAgICAgICAgICAgICAgICAgICAgICAgICAgICBEZXJlY2F0ZWQgSmF2YSBzdHlsZSBjb25zdGFudHMgKGVnLCBJTlRFR0VSX0ZJRUxEKQoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgoqLwogCiNpZm5kZWYgTlVNRk1UX0gKI2RlZmluZSBOVU1GTVRfSAogCgojaW5jbHVkZSAidW5pY29kZS91dHlwZXMuaCIKI2luY2x1ZGUgInVuaWNvZGUvdW5pc3RyLmgiCiNpbmNsdWRlICJ1bmljb2RlL2Zvcm1hdC5oIgoKY2xhc3MgTG9jYWxlOwoKLyoqCiAqIEFic3RyYWN0IGJhc2UgY2xhc3MgZm9yIGFsbCBudW1iZXIgZm9ybWF0cy4gIFByb3ZpZGVzIGludGVyZmFjZSBmb3IKICogZm9ybWF0dGluZyBhbmQgcGFyc2luZyBhIG51bWJlci4gIEFsc28gcHJvdmlkZXMgbWV0aG9kcyBmb3IKICogZGV0ZXJtaW5pbmcgd2hpY2ggbG9jYWxlcyBoYXZlIG51bWJlciBmb3JtYXRzLCBhbmQgd2hhdCB0aGVpciBuYW1lcwogKiBhcmUuCiAqIDxQPgogKiBOdW1iZXJGb3JtYXQgaGVscHMgeW91IHRvIGZvcm1hdCBhbmQgcGFyc2UgbnVtYmVycyBmb3IgYW55IGxvY2FsZS4KICogWW91ciBjb2RlIGNhbiBiZSBjb21wbGV0ZWx5IGluZGVwZW5kZW50IG9mIHRoZSBsb2NhbGUgY29udmVudGlvbnMKICogZm9yIGRlY2ltYWwgcG9pbnRzLCB0aG91c2FuZHMtc2VwYXJhdG9ycywgb3IgZXZlbiB0aGUgcGFydGljdWxhcgogKiBkZWNpbWFsIGRpZ2l0cyB1c2VkLCBvciB3aGV0aGVyIHRoZSBudW1iZXIgZm9ybWF0IGlzIGV2ZW4gZGVjaW1hbC4KICogPFA+CiAqIFRvIGZvcm1hdCBhIG51bWJlciBmb3IgdGhlIGN1cnJlbnQgTG9jYWxlLCB1c2Ugb25lIG9mIHRoZSBzdGF0aWMKICogZmFjdG9yeSBtZXRob2RzOgogKiA8cHJlPgogKiAuICAgZG91YmxlIG15TnVtYmVyID0gNy4wOwogKiAuICAgVW5pY29kZVN0cmluZyBteVN0cmluZzsKICogLiAgIFVFcnJvckNvZGUgc3VjY2VzcyA9IFVfWkVST19FUlJPUjsKICogLiAgIE51bWJlckZvcm1hdCogbmYgPSBOdW1iZXJGb3JtYXQ6OmNyZWF0ZUluc3RhbmNlKHN1Y2Nlc3MpCiAqIC4gICBuZi0+Zm9ybWF0KG15TnVtYmVyLCBteVN0cmluZyk7CiAqIC4gICBjb3V0ICZsdDsmbHQ7ICIgRXhhbXBsZSAxOiAiICZsdDsmbHQ7IG15U3RyaW5nICZsdDsmbHQ7IGVuZGw7CiAqIDwvcHJlPgogKiBJZiB5b3UgYXJlIGZvcm1hdHRpbmcgbXVsdGlwbGUgbnVtYmVycywgaXQgaXMgbW9yZSBlZmZpY2llbnQgdG8gZ2V0CiAqIHRoZSBmb3JtYXQgYW5kIHVzZSBpdCBtdWx0aXBsZSB0aW1lcyBzbyB0aGF0IHRoZSBzeXN0ZW0gZG9lc24ndAogKiBoYXZlIHRvIGZldGNoIHRoZSBpbmZvcm1hdGlvbiBhYm91dCB0aGUgbG9jYWwgbGFuZ3VhZ2UgYW5kIGNvdW50cnkKICogY29udmVudGlvbnMgbXVsdGlwbGUgdGltZXMuCiAqIDxwcmU+CiAqIC4gICAgVW5pY29kZVN0cmluZyBteVN0cmluZzsKICogLiAgICBVRXJyb3JDb2RlIHN1Y2Nlc3MgPSBVX1pFUk9fRVJST1I7CiAqIC4gICAgbmYgPSBOdW1iZXJGb3JtYXQ6OmNyZWF0ZUluc3RhbmNlKCBzdWNjZXNzICk7CiAqIC4gICAgaW50MzJfdCBhW10gPSB7IDEyMywgMzMzMywgLTEyMzQ1NjcgfTsKICogLiAgICBjb25zdCBpbnQzMl90IGFfbGVuID0gc2l6ZW9mKGEpIC8gc2l6ZW9mKGFbMF0pOwogKiAuICAgIG15U3RyaW5nLnJlbW92ZSgpOwogKiAuICAgIGZvciAoaW50MzJfdCBpID0gMDsgaSA8IGFfbGVuOyBpKyspIHsKICogLiAgICAgICAgbmYtPmZvcm1hdChhW2ldLCBteVN0cmluZyk7CiAqIC4gICAgICAgIG15U3RyaW5nICs9ICIgOyAiOwogKiAuICAgIH0KICogLiAgICBjb3V0ICZsdDsmbHQ7ICIgRXhhbXBsZSAyOiAiICZsdDsmbHQ7IG15U3RyaW5nICZsdDsmbHQ7IGVuZGw7CiAqIDwvcHJlPgogKiBUbyBmb3JtYXQgYSBudW1iZXIgZm9yIGEgZGlmZmVyZW50IExvY2FsZSwgc3BlY2lmeSBpdCBpbiB0aGUKICogY2FsbCB0byBjcmVhdGVJbnN0YW5jZSgpLgogKiA8cHJlPgogKiAuICAgIG5mID0gTnVtYmVyRm9ybWF0OjpjcmVhdGVJbnN0YW5jZSggTG9jYWxlOjpGUkVOQ0gsIHN1Y2Nlc3MgKTsKICogPC9wcmU+CiAqIFlvdSBjYW4gdXNlIGEgTnVtYmVyRm9ybWF0IHRvIHBhcnNlIGFsc28uCiAqIDxwcmU+CiAqIC4gICBVRXJyb3JDb2RlIHN1Y2Nlc3M7CiAqIC4gICBGb3JtYXR0YWJsZSByZXN1bHQoLTk5OSk7ICAvLyBpbml0aWFsaXplZCB3aXRoIGVycm9yIGNvZGUKICogLiAgIG5mLT5wYXJzZShteVN0cmluZywgcmVzdWx0LCBzdWNjZXNzKTsKICogPC9wcmU+CiAqIFVzZSBjcmVhdGVJbnN0YW5jZSB0byBnZXQgdGhlIG5vcm1hbCBudW1iZXIgZm9ybWF0IGZvciB0aGF0IGNvdW50cnkuCiAqIFRoZXJlIGFyZSBvdGhlciBzdGF0aWMgZmFjdG9yeSBtZXRob2RzIGF2YWlsYWJsZS4gIFVzZSBnZXRDdXJyZW5jeQogKiB0byBnZXQgdGhlIGN1cnJlbmN5IG51bWJlciBmb3JtYXQgZm9yIHRoYXQgY291bnRyeS4gIFVzZSBnZXRQZXJjZW50CiAqIHRvIGdldCBhIGZvcm1hdCBmb3IgZGlzcGxheWluZyBwZXJjZW50YWdlcy4gV2l0aCB0aGlzIGZvcm1hdCwgYQogKiBmcmFjdGlvbiBmcm9tIDAuNTMgaXMgZGlzcGxheWVkIGFzIDUzJS4KICogPFA+CiAqIFlvdSBjYW4gYWxzbyBjb250cm9sIHRoZSBkaXNwbGF5IG9mIG51bWJlcnMgd2l0aCBzdWNoIG1ldGhvZHMgYXMKICogZ2V0TWluaW11bUZyYWN0aW9uRGlnaXRzLiAgSWYgeW91IHdhbnQgZXZlbiBtb3JlIGNvbnRyb2wgb3ZlciB0aGUKICogZm9ybWF0IG9yIHBhcnNpbmcsIG9yIHdhbnQgdG8gZ2l2ZSB5b3VyIHVzZXJzIG1vcmUgY29udHJvbCwgeW91IGNhbgogKiB0cnkgY2FzdGluZyB0aGUgTnVtYmVyRm9ybWF0IHlvdSBnZXQgZnJvbSB0aGUgZmFjdG9yeSBtZXRob2RzIHRvIGEKICogRGVjaW1hbE51bWJlckZvcm1hdC4gVGhpcyB3aWxsIHdvcmsgZm9yIHRoZSB2YXN0IG1ham9yaXR5IG9mCiAqIGNvdW50cmllczsganVzdCByZW1lbWJlciB0byBwdXQgaXQgaW4gYSB0cnkgYmxvY2sgaW4gY2FzZSB5b3UKICogZW5jb3VudGVyIGFuIHVudXN1YWwgb25lLgogKiA8UD4KICogWW91IGNhbiBhbHNvIHVzZSBmb3JtcyBvZiB0aGUgcGFyc2UgYW5kIGZvcm1hdCBtZXRob2RzIHdpdGgKICogUGFyc2VQb3NpdGlvbiBhbmQgRmllbGRQb3NpdGlvbiB0byBhbGxvdyB5b3UgdG86CiAqIDx1bCB0eXBlPXJvdW5kPgogKiAgIDxsaT4oYSkgcHJvZ3Jlc3NpdmVseSBwYXJzZSB0aHJvdWdoIHBpZWNlcyBvZiBhIHN0cmluZy4KICogICA8bGk+KGIpIGFsaWduIHRoZSBkZWNpbWFsIHBvaW50IGFuZCBvdGhlciBhcmVhcy4KICogPC91bD4KICogRm9yIGV4YW1wbGUsIHlvdSBjYW4gYWxpZ24gbnVtYmVycyBpbiB0d28gd2F5cy4KICogPFA+CiAqIElmIHlvdSBhcmUgdXNpbmcgYSBtb25vc3BhY2VkIGZvbnQgd2l0aCBzcGFjaW5nIGZvciBhbGlnbm1lbnQsIHlvdQogKiBjYW4gcGFzcyB0aGUgRmllbGRQb3NpdGlvbiBpbiB5b3VyIGZvcm1hdCBjYWxsLCB3aXRoIGZpZWxkID0KICogSU5URUdFUl9GSUVMRC4gT24gb3V0cHV0LCBnZXRFbmRJbmRleCB3aWxsIGJlIHNldCB0byB0aGUgb2Zmc2V0CiAqIGJldHdlZW4gdGhlIGxhc3QgY2hhcmFjdGVyIG9mIHRoZSBpbnRlZ2VyIGFuZCB0aGUgZGVjaW1hbC4gQWRkCiAqIChkZXNpcmVkU3BhY2VDb3VudCAtIGdldEVuZEluZGV4KSBzcGFjZXMgYXQgdGhlIGZyb250IG9mIHRoZQogKiBzdHJpbmcuCiAqIDxQPgogKiBJZiB5b3UgYXJlIHVzaW5nIHByb3BvcnRpb25hbCBmb250cywgaW5zdGVhZCBvZiBwYWRkaW5nIHdpdGgKICogc3BhY2VzLCBtZWFzdXJlIHRoZSB3aWR0aCBvZiB0aGUgc3RyaW5nIGluIHBpeGVscyBmcm9tIHRoZSBzdGFydCB0bwogKiBnZXRFbmRJbmRleC4gIFRoZW4gbW92ZSB0aGUgcGVuIGJ5IChkZXNpcmVkUGl4ZWxXaWR0aCAtCiAqIHdpZHRoVG9BbGlnbm1lbnRQb2ludCkgYmVmb3JlIGRyYXdpbmcgdGhlIHRleHQuICBJdCBhbHNvIHdvcmtzCiAqIHdoZXJlIHRoZXJlIGlzIG5vIGRlY2ltYWwsIGJ1dCBwb3NzaWJseSBhZGRpdGlvbmFsIGNoYXJhY3RlcnMgYXQKICogdGhlIGVuZCwgZS5nLiB3aXRoIHBhcmVudGhlc2VzIGluIG5lZ2F0aXZlIG51bWJlcnM6ICIoMTIpIiBmb3IgLTEyLgogKiBAc3RhYmxlCiAqLwpjbGFzcyBVX0kxOE5fQVBJIE51bWJlckZvcm1hdCA6IHB1YmxpYyBGb3JtYXQgewpwdWJsaWM6CgogICAgLyoqCiAgICAgKiBBbGlnbm1lbnQgRmllbGQgY29uc3RhbnRzIHVzZWQgdG8gY29uc3RydWN0IGEgRmllbGRQb3NpdGlvbiBvYmplY3QuCiAgICAgKiBTaWduaWZpZXMgdGhhdCB0aGUgcG9zaXRpb24gb2YgdGhlIGludGVnZXIgcGFydCBvciBmcmFjdGlvbiBwYXJ0IG9mCiAgICAgKiBhIGZvcm1hdHRlZCBudW1iZXIgc2hvdWxkIGJlIHJldHVybmVkLgogICAgICoKICAgICAqIEBzZWUgRmllbGRQb3NpdGlvbgogICAgICogQHN0YWJsZQogICAgICovCiAgICBlbnVtIEVBbGlnbm1lbnRGaWVsZHMgewogICAgICAgIGtJbnRlZ2VyRmllbGQsCiAgICAgICAga0ZyYWN0aW9uRmllbGQsCgoKICAgIC8qKgogICAgICogVGhlc2UgY29uc3RhbnRzIGFyZSBwcm92aWRlZCBmb3IgYmFja3dhcmRzIGNvbXBhdGliaWxpdHkgb25seSwKICAgICAqIGFuZCBhcmUgZGVwcmVjYXRlZC4gIFBsZWFzZSB1c2UgdGhlIEMrKyBzdHlsZSBjb25zdGFudHMgZGVmaW5lZCBhYm92ZS4KICAgICAqIEBzdGFibGUKICAgICAqLyAgICAgICAKICAgICAgICBJTlRFR0VSX0ZJRUxEICAgICAgICA9IGtJbnRlZ2VyRmllbGQsCiAgICAgICAgRlJBQ1RJT05fRklFTEQgICAgICAgID0ga0ZyYWN0aW9uRmllbGQKICAgIH07CgogICAgdmlydHVhbCB+TnVtYmVyRm9ybWF0KCk7CgogICAgLyoqCiAgICAgKiBSZXR1cm4gdHJ1ZSBpZiB0aGUgZ2l2ZW4gRm9ybWF0IG9iamVjdHMgYXJlIHNlbWFudGljYWxseSBlcXVhbC4KICAgICAqIE9iamVjdHMgb2YgZGlmZmVyZW50IHN1YmNsYXNzZXMgYXJlIGNvbnNpZGVyZWQgdW5lcXVhbC4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBib29sX3Qgb3BlcmF0b3I9PShjb25zdCBGb3JtYXQmIG90aGVyKSBjb25zdDsKCiAgICAvKioKICAgICAqIEZvcm1hdCBhbiBvYmplY3QgdG8gcHJvZHVjZSBhIHN0cmluZy4gIFRoaXMgbWV0aG9kIGhhbmRsZXMKICAgICAqIEZvcm1hdHRhYmxlIG9iamVjdHMgd2l0aCBudW1lcmljIHR5cGVzLiBJZiB0aGUgRm9ybWF0dGFibGUKICAgICAqIG9iamVjdCB0eXBlIGlzIG5vdCBhIG51bWVyaWMgdHlwZSwgdGhlbiBpdCByZXR1cm5zIGEgZmFpbGluZwogICAgICogVUVycm9yQ29kZS4KICAgICAqCiAgICAgKiBAcGFyYW0gb2JqICAgICAgICAgICBUaGUgb2JqZWN0IHRvIGZvcm1hdC4KICAgICAqIEBwYXJhbSB0b0FwcGVuZFRvICAgIFdoZXJlIHRoZSB0ZXh0IGlzIHRvIGJlIGFwcGVuZGVkLgogICAgICogQHBhcmFtIHBvcyAgICAgICAgICAgT24gaW5wdXQ6IGFuIGFsaWdubWVudCBmaWVsZCwgaWYgZGVzaXJlZC4KICAgICAqICAgICAgICAgICAgICAgICAgICAgIE9uIG91dHB1dDogdGhlIG9mZnNldHMgb2YgdGhlIGFsaWdubWVudCBmaWVsZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgICAgIE91dHB1dCBwYXJhbSBmaWxsZWQgd2l0aCBzdWNjZXNzL2ZhaWx1cmUgc3RhdHVzLgogICAgICogQHJldHVybiAgICAgICAgICAgICAgVGhlIHZhbHVlIHBhc3NlZCBpbiBhcyB0b0FwcGVuZFRvICh0aGlzIGFsbG93cyBjaGFpbmluZywKICAgICAqICAgICAgICAgICAgICAgICAgICAgIGFzIHdpdGggVW5pY29kZVN0cmluZzo6YXBwZW5kKCkpCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgVW5pY29kZVN0cmluZyYgZm9ybWF0KGNvbnN0IEZvcm1hdHRhYmxlJiBvYmosCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiB0b0FwcGVuZFRvLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRmllbGRQb3NpdGlvbiYgcG9zLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdDsKCiAgICAvKioKICAgICAqIFBhcnNlIGEgc3RyaW5nIHRvIHByb2R1Y2UgYW4gb2JqZWN0LiAgVGhpcyBtZXRob2RzIGhhbmRsZXMKICAgICAqIHBhcnNpbmcgb2YgbnVtZXJpYyBzdHJpbmdzIGludG8gRm9ybWF0dGFibGUgb2JqZWN0cyB3aXRoIG51bWVyaWMKICAgICAqIHR5cGVzLgogICAgICogPFA+CiAgICAgKiBCZWZvcmUgY2FsbGluZywgc2V0IHBhcnNlX3Bvcy5pbmRleCB0byB0aGUgb2Zmc2V0IHlvdSB3YW50IHRvCiAgICAgKiBzdGFydCBwYXJzaW5nIGF0IGluIHRoZSBzb3VyY2UuIEFmdGVyIGNhbGxpbmcsIHBhcnNlX3Bvcy5pbmRleAogICAgICogaXMgdGhlIGVuZCBvZiB0aGUgdGV4dCB5b3UgcGFyc2VkLiAgSWYgZXJyb3Igb2NjdXJzLCBpbmRleCBpcwogICAgICogdW5jaGFuZ2VkLgogICAgICogPFA+CiAgICAgKiBXaGVuIHBhcnNpbmcsIGxlYWRpbmcgd2hpdGVzcGFjZSBpcyBkaXNjYXJkZWQgKHdpdGggc3VjY2Vzc2Z1bAogICAgICogcGFyc2UpLCB3aGlsZSB0cmFpbGluZyB3aGl0ZXNwYWNlIGlzIGxlZnQgYXMgaXMuCiAgICAgKiA8UD4KICAgICAqIFNlZSBGb3JtYXQ6OnBhcnNlT2JqZWN0KCkgZm9yIG1vcmUuCiAgICAgKgogICAgICogQHBhcmFtIHNvdXJjZSAgICBUaGUgc3RyaW5nIHRvIGJlIHBhcnNlZCBpbnRvIGFuIG9iamVjdC4KICAgICAqIEBwYXJhbSByZXN1bHQgICAgRm9ybWF0dGFibGUgdG8gYmUgc2V0IHRvIHRoZSBwYXJzZSByZXN1bHQuCiAgICAgKiAgICAgICAgICAgICAgICAgIElmIHBhcnNlIGZhaWxzLCByZXR1cm4gY29udGVudHMgYXJlIHVuZGVmaW5lZC4KICAgICAqIEBwYXJhbSBwYXJzZV9wb3MgVGhlIHBvc2l0aW9uIHRvIHN0YXJ0IHBhcnNpbmcgYXQuIFVwb24gcmV0dXJuCiAgICAgKiAgICAgICAgICAgICAgICAgIHRoaXMgcGFyYW0gaXMgc2V0IHRvIHRoZSBwb3NpdGlvbiBhZnRlciB0aGUKICAgICAqICAgICAgICAgICAgICAgICAgbGFzdCBjaGFyYWN0ZXIgc3VjY2Vzc2Z1bGx5IHBhcnNlZC4gSWYgdGhlCiAgICAgKiAgICAgICAgICAgICAgICAgIHNvdXJjZSBpcyBub3QgcGFyc2VkIHN1Y2Nlc3NmdWxseSwgdGhpcyBwYXJhbQogICAgICogICAgICAgICAgICAgICAgICB3aWxsIHJlbWFpbiB1bmNoYW5nZWQuCiAgICAgKiBAcmV0dXJuICAgICAgICAgIEEgbmV3bHkgY3JlYXRlZCBGb3JtYXR0YWJsZSogb2JqZWN0LCBvciBOVUxMCiAgICAgKiAgICAgICAgICAgICAgICAgIG9uIGZhaWx1cmUuICBUaGUgY2FsbGVyIG93bnMgdGhpcyBhbmQgc2hvdWxkCiAgICAgKiAgICAgICAgICAgICAgICAgIGRlbGV0ZSBpdCB3aGVuIGRvbmUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBwYXJzZU9iamVjdChjb25zdCBVbmljb2RlU3RyaW5nJiBzb3VyY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRm9ybWF0dGFibGUmIHJlc3VsdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJzZVBvc2l0aW9uJiBwYXJzZV9wb3MpIGNvbnN0OwoKICAgIC8qKgogICAgICogRm9ybWF0IGEgZG91YmxlIG9yIGxvbmcgbnVtYmVyLiBUaGVzZSBtZXRob2RzIGNhbGwgdGhlIE51bWJlckZvcm1hdAogICAgICogcHVyZSB2aXJ0dWFsIGZvcm1hdCgpIG1ldGhvZHMgd2l0aCB0aGUgZGVmYXVsdCBGaWVsZFBvc2l0aW9uLgogICAgICoKICAgICAqIEBwYXJhbSBudW1iZXIgICAgVGhlIHZhbHVlIHRvIGJlIGZvcm1hdHRlZC4KICAgICAqIEBwYXJhbSBvdXRwdXQgICAgT3V0cHV0IHBhcmFtIHdpdGggdGhlIGZvcm1hdHRlZCBzdHJpbmcuCiAgICAgKiBAcmV0dXJuICAgICAgICAgIEEgcmVmZXJlbmNlIHRvICdvdXRwdXQnIHBhcmFtLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBVbmljb2RlU3RyaW5nJiBmb3JtYXQoICBkb3VibGUgbnVtYmVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgb3V0cHV0KSBjb25zdDsKCiAgICBVbmljb2RlU3RyaW5nJiBmb3JtYXQoICBpbnQzMl90IG51bWJlciwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIG91dHB1dCkgY29uc3Q7CgogICAvKioKICAgICogRm9ybWF0IGEgZG91YmxlIG9yIGxvbmcgbnVtYmVyLiBDb25jcmV0ZSBzdWJjbGFzc2VzIG11c3QgaW1wbGVtZW50CiAgICAqIHRoZXNlIHB1cmUgdmlydHVhbCBtZXRob2RzLgogICAgKgogICAgKiBAcGFyYW0gbnVtYmVyICAgICBUaGUgdmFsdWUgdG8gYmUgZm9ybWF0dGVkLgogICAgKiBAcGFyYW0gdG9BcHBlbmRUbyBUaGUgc3RyaW5nIHRvIGFwcGVuZCB0aGUgZm9ybWF0dGVkIHN0cmluZyB0by4KICAgICogICAgICAgICAgICAgICAgICAgVGhpcyBpcyBhbiBvdXRwdXQgcGFyYW1ldGVyLgogICAgKiBAcGFyYW0gcG9zICAgICAgICBPbiBpbnB1dDogYW4gYWxpZ25tZW50IGZpZWxkLCBpZiBkZXNpcmVkLgogICAgKiAgICAgICAgICAgICAgICAgICBPbiBvdXRwdXQ6IHRoZSBvZmZzZXRzIG9mIHRoZSBhbGlnbm1lbnQgZmllbGQuCiAgICAqIEByZXR1cm4gICAgICAgICAgIEEgcmVmZXJlbmNlIHRvICd0b0FwcGVuZFRvJy4KICAgICAqIEBzdGFibGUKICAgICovCiAgICB2aXJ0dWFsIFVuaWNvZGVTdHJpbmcmIGZvcm1hdChkb3VibGUgbnVtYmVyLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgdG9BcHBlbmRUbywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZpZWxkUG9zaXRpb24mIHBvcykgY29uc3QgPSAwOwogICAgdmlydHVhbCBVbmljb2RlU3RyaW5nJiBmb3JtYXQoaW50MzJfdCBudW1iZXIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiB0b0FwcGVuZFRvLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRmllbGRQb3NpdGlvbiYgcG9zKSBjb25zdCA9IDA7CgogICAgLyoqCiAgICAgKiBSZWRlY2xhcmVkIEZvcm1hdCBtZXRob2QuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIFVuaWNvZGVTdHJpbmcmIGZvcm1hdChjb25zdCBGb3JtYXR0YWJsZSYgb2JqLAogICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIHJlc3VsdCwKICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0OwoKICAgLyoqCiAgICAqIFJldHVybiBhIGxvbmcgaWYgcG9zc2libGUgKGUuZy4gd2l0aGluIHJhbmdlIExPTkdfTUFYLAogICAgKiBMT05HX01BWF0sIGFuZCB3aXRoIG5vIGRlY2ltYWxzKSwgb3RoZXJ3aXNlIGEgZG91YmxlLiAgSWYKICAgICogSW50ZWdlck9ubHkgaXMgc2V0LCB3aWxsIHN0b3AgYXQgYSBkZWNpbWFsIHBvaW50IChvciBlcXVpdmFsZW50OwogICAgKiBlLmcuIGZvciByYXRpb25hbCBudW1iZXJzICIxIDIvMyIsIHdpbGwgc3RvcCBhZnRlciB0aGUgMSkuCiAgICAqIDxQPgogICAgKiBJZiBubyBvYmplY3QgY2FuIGJlIHBhcnNlZCwgaW5kZXggaXMgdW5jaGFuZ2VkLCBhbmQgTlVMTCBpcwogICAgKiByZXR1cm5lZC4KICAgICogPFA+CiAgICAqIFRoaXMgaXMgYSBwdXJlIHZpcnR1YWwgd2hpY2ggY29uY3JldGUgc3ViY2xhc3NlcyBtdXN0IGltcGxlbWVudC4KICAgICoKICAgICogQHBhcmFtIHRleHQgICAgICAgICAgIFRoZSB0ZXh0IHRvIGJlIHBhcnNlZC4KICAgICogQHBhcmFtIHJlc3VsdCAgICAgICAgIEZvcm1hdHRhYmxlIHRvIGJlIHNldCB0byB0aGUgcGFyc2UgcmVzdWx0LgogICAgKiAgICAgICAgICAgICAgICAgICAgICAgSWYgcGFyc2UgZmFpbHMsIHJldHVybiBjb250ZW50cyBhcmUgdW5kZWZpbmVkLgogICAgKiBAcGFyYW0gcGFyc2VQb3NpdGlvbiAgVGhlIHBvc2l0aW9uIHRvIHN0YXJ0IHBhcnNpbmcgYXQgb24gaW5wdXQuCiAgICAqICAgICAgICAgICAgICAgICAgICAgICBPbiBvdXRwdXQsIG1vdmVkIHRvIGFmdGVyIHRoZSBsYXN0IHN1Y2Nlc3NmdWxseQogICAgKiAgICAgICAgICAgICAgICAgICAgICAgcGFyc2UgY2hhcmFjdGVyLiBPbiBwYXJzZSBmYWlsdXJlLCBkb2VzIG5vdCBjaGFuZ2UuCiAgICAqIEByZXR1cm4gICAgICAgICAgICAgICBBIEZvcm1hdHRhYmxlIG9iamVjdCBvZiBudW1lcmljIHR5cGUuICBUaGUgY2FsbGVyCiAgICAqICAgICAgICAgICAgICAgICAgICAgICBvd25zIHRoaXMgYW4gbXVzdCBkZWxldGUgaXQuICBOVUxMIG9uIGZhaWx1cmUuCiAgICAqIEBzdGFibGUKICAgICovCiAgICB2aXJ0dWFsIHZvaWQgcGFyc2UoY29uc3QgVW5pY29kZVN0cmluZyYgdGV4dCwKICAgICAgICAgICAgICAgICAgICAgICBGb3JtYXR0YWJsZSYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgIFBhcnNlUG9zaXRpb24mIHBhcnNlUG9zaXRpb24pIGNvbnN0ID0gMDsKCiAgICAvKioKICAgICAqIFBhcnNlIGEgc3RyaW5nIGFzIGEgbnVtZXJpYyB2YWx1ZSwgYW5kIHJldHVybiBhIEZvcm1hdHRhYmxlCiAgICAgKiBudW1lcmljIG9iamVjdC4gVGhpcyBtZXRob2QgcGFyc2VzIGludGVnZXJzIG9ubHkgaWYgSW50ZWdlck9ubHkKICAgICAqIGlzIHNldC4KICAgICAqCiAgICAgKiBAcGFyYW0gdGV4dCAgICAgICAgICBUaGUgdGV4dCB0byBiZSBwYXJzZWQuCiAgICAgKiBAcGFyYW0gcmVzdWx0ICAgICAgICBGb3JtYXR0YWJsZSB0byBiZSBzZXQgdG8gdGhlIHBhcnNlIHJlc3VsdC4KICAgICAqICAgICAgICAgICAgICAgICAgICAgIElmIHBhcnNlIGZhaWxzLCByZXR1cm4gY29udGVudHMgYXJlIHVuZGVmaW5lZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgICAgIFN1Y2Nlc3Mgb3IgZmFpbHVyZSBvdXRwdXQgcGFyYW1ldGVyLgogICAgICogQHJldHVybiAgICAgICAgICAgICAgQSBGb3JtYXR0YWJsZSBvYmplY3Qgb2YgbnVtZXJpYyB0eXBlLiAgVGhlIGNhbGxlcgogICAgICogICAgICAgICAgICAgICAgICAgICAgb3ducyB0aGlzIGFuIG11c3QgZGVsZXRlIGl0LiAgTlVMTCBvbiBmYWlsdXJlLgogICAgICogQHNlZSAgICAgICAgICAgICAgICAgTnVtYmVyRm9ybWF0Ojppc1BhcnNlSW50ZWdlck9ubHkKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHBhcnNlKCBjb25zdCBVbmljb2RlU3RyaW5nJiB0ZXh0LAogICAgICAgICAgICAgICAgICAgICAgICBGb3JtYXR0YWJsZSYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0OwoKICAgIC8qKgogICAgICogUmV0dXJuIHRydWUgaWYgdGhpcyBmb3JtYXQgd2lsbCBwYXJzZSBudW1iZXJzIGFzIGludGVnZXJzCiAgICAgKiBvbmx5LiAgRm9yIGV4YW1wbGUgaW4gdGhlIEVuZ2xpc2ggbG9jYWxlLCB3aXRoIFBhcnNlSW50ZWdlck9ubHkKICAgICAqIHRydWUsIHRoZSBzdHJpbmcgIjEyMzQuIiB3b3VsZCBiZSBwYXJzZWQgYXMgdGhlIGludGVnZXIgdmFsdWUKICAgICAqIDEyMzQgYW5kIHBhcnNpbmcgd291bGQgc3RvcCBhdCB0aGUgIi4iIGNoYXJhY3Rlci4gIE9mIGNvdXJzZSwKICAgICAqIHRoZSBleGFjdCBmb3JtYXQgYWNjZXB0ZWQgYnkgdGhlIHBhcnNlIG9wZXJhdGlvbiBpcyBsb2NhbGUKICAgICAqIGRlcGVuZGFudCBhbmQgZGV0ZXJtaW5lZCBieSBzdWItY2xhc3NlcyBvZiBOdW1iZXJGb3JtYXQuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIGJvb2xfdCBpc1BhcnNlSW50ZWdlck9ubHkodm9pZCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBTZXRzIHdoZXRoZXIgb3Igbm90IG51bWJlcnMgc2hvdWxkIGJlIHBhcnNlZCBhcyBpbnRlZ2VycyBvbmx5LgogICAgICogQHNlZSBpc1BhcnNlSW50ZWdlck9ubHkKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldFBhcnNlSW50ZWdlck9ubHkoYm9vbF90IHZhbHVlKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgdGhlIGRlZmF1bHQgbnVtYmVyIGZvcm1hdCBmb3IgdGhlIGN1cnJlbnQgZGVmYXVsdAogICAgICogbG9jYWxlLiAgVGhlIGRlZmF1bHQgZm9ybWF0IGlzIG9uZSBvZiB0aGUgc3R5bGVzIHByb3ZpZGVkIGJ5CiAgICAgKiB0aGUgb3RoZXIgZmFjdG9yeSBtZXRob2RzOiBnZXROdW1iZXJJbnN0YW5jZSwKICAgICAqIGdldEN1cnJlbmN5SW5zdGFuY2Ugb3IgZ2V0UGVyY2VudEluc3RhbmNlLiAgRXhhY3RseSB3aGljaCBvbmUKICAgICAqIGlzIGxvY2FsZSBkZXBlbmRhbnQuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGNyZWF0ZUluc3RhbmNlKFVFcnJvckNvZGUmKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgdGhlIGRlZmF1bHQgbnVtYmVyIGZvcm1hdCBmb3IgdGhlIHNwZWNpZmllZCBsb2NhbGUuCiAgICAgKiBUaGUgZGVmYXVsdCBmb3JtYXQgaXMgb25lIG9mIHRoZSBzdHlsZXMgcHJvdmlkZWQgYnkgdGhlIG90aGVyCiAgICAgKiBmYWN0b3J5IG1ldGhvZHM6IGdldE51bWJlckluc3RhbmNlLCBnZXRDdXJyZW5jeUluc3RhbmNlIG9yCiAgICAgKiBnZXRQZXJjZW50SW5zdGFuY2UuICBFeGFjdGx5IHdoaWNoIG9uZSBpcyBsb2NhbGUgZGVwZW5kYW50LgogICAgICogQHN0YWJsZQogICAgICovCiAgICBzdGF0aWMgTnVtYmVyRm9ybWF0KiBjcmVhdGVJbnN0YW5jZShjb25zdCBMb2NhbGUmIGluTG9jYWxlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYpOwoKICAgIC8qKgogICAgICogUmV0dXJucyBhIGN1cnJlbmN5IGZvcm1hdCBmb3IgdGhlIGN1cnJlbnQgZGVmYXVsdCBsb2NhbGUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGNyZWF0ZUN1cnJlbmN5SW5zdGFuY2UoVUVycm9yQ29kZSYpOwoKICAgIC8qKgogICAgICogUmV0dXJucyBhIGN1cnJlbmN5IGZvcm1hdCBmb3IgdGhlIHNwZWNpZmllZCBsb2NhbGUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGNyZWF0ZUN1cnJlbmN5SW5zdGFuY2UoY29uc3QgTG9jYWxlJiBpbkxvY2FsZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYpOwoKICAgIC8qKgogICAgICogUmV0dXJucyBhIHBlcmNlbnRhZ2UgZm9ybWF0IGZvciB0aGUgY3VycmVudCBkZWZhdWx0IGxvY2FsZS4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgc3RhdGljIE51bWJlckZvcm1hdCogY3JlYXRlUGVyY2VudEluc3RhbmNlKFVFcnJvckNvZGUmKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgYSBwZXJjZW50YWdlIGZvcm1hdCBmb3IgdGhlIHNwZWNpZmllZCBsb2NhbGUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGNyZWF0ZVBlcmNlbnRJbnN0YW5jZShjb25zdCBMb2NhbGUmIGluTG9jYWxlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgYSBzY2llbnRpZmljIGZvcm1hdCBmb3IgdGhlIGN1cnJlbnQgZGVmYXVsdCBsb2NhbGUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGNyZWF0ZVNjaWVudGlmaWNJbnN0YW5jZShVRXJyb3JDb2RlJik7CgogICAgLyoqCiAgICAgKiBSZXR1cm5zIGEgc2NpZW50aWZpYyBmb3JtYXQgZm9yIHRoZSBzcGVjaWZpZWQgbG9jYWxlLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBzdGF0aWMgTnVtYmVyRm9ybWF0KiBjcmVhdGVTY2llbnRpZmljSW5zdGFuY2UoY29uc3QgTG9jYWxlJiBpbkxvY2FsZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYpOwoKICAgIC8qKgogICAgICogR2V0IHRoZSBzZXQgb2YgTG9jYWxlcyBmb3Igd2hpY2ggTnVtYmVyRm9ybWF0cyBhcmUgaW5zdGFsbGVkLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBzdGF0aWMgY29uc3QgTG9jYWxlKiBnZXRBdmFpbGFibGVMb2NhbGVzKGludDMyX3QmIGNvdW50KTsKCiAgICAvKioKICAgICAqIFJldHVybnMgdHJ1ZSBpZiBncm91cGluZyBpcyB1c2VkIGluIHRoaXMgZm9ybWF0LiBGb3IgZXhhbXBsZSwKICAgICAqIGluIHRoZSBFbmdsaXNoIGxvY2FsZSwgd2l0aCBncm91cGluZyBvbiwgdGhlIG51bWJlciAxMjM0NTY3CiAgICAgKiBtaWdodCBiZSBmb3JtYXR0ZWQgYXMgIjEsMjM0LDU2NyIuIFRoZSBncm91cGluZyBzZXBhcmF0b3IgYXMKICAgICAqIHdlbGwgYXMgdGhlIHNpemUgb2YgZWFjaCBncm91cCBpcyBsb2NhbGUgZGVwZW5kYW50IGFuZCBpcwogICAgICogZGV0ZXJtaW5lZCBieSBzdWItY2xhc3NlcyBvZiBOdW1iZXJGb3JtYXQuCiAgICAgKiBAc2VlIHNldEdyb3VwaW5nVXNlZAogICAgICogQHN0YWJsZQogICAgICovCiAgICBib29sX3QgaXNHcm91cGluZ1VzZWQodm9pZCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBTZXQgd2hldGhlciBvciBub3QgZ3JvdXBpbmcgd2lsbCBiZSB1c2VkIGluIHRoaXMgZm9ybWF0LgogICAgICogQHNlZSBnZXRHcm91cGluZ1VzZWQKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldEdyb3VwaW5nVXNlZChib29sX3QgbmV3VmFsdWUpOwoKICAgIC8qKgogICAgICogUmV0dXJucyB0aGUgbWF4aW11bSBudW1iZXIgb2YgZGlnaXRzIGFsbG93ZWQgaW4gdGhlIGludGVnZXIgcG9ydGlvbiBvZiBhCiAgICAgKiBudW1iZXIuCiAgICAgKiBAc2VlIHNldE1heGltdW1JbnRlZ2VyRGlnaXRzCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIGludDMyX3QgZ2V0TWF4aW11bUludGVnZXJEaWdpdHModm9pZCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBTZXRzIHRoZSBtYXhpbXVtIG51bWJlciBvZiBkaWdpdHMgYWxsb3dlZCBpbiB0aGUgaW50ZWdlciBwb3J0aW9uIG9mIGEKICAgICAqIG51bWJlci4gbWF4aW11bUludGVnZXJEaWdpdHMgbXVzdCBiZSA+PSBtaW5pbXVtSW50ZWdlckRpZ2l0cy4gIElmIHRoZQogICAgICogbmV3IHZhbHVlIGZvciBtYXhpbXVtSW50ZWdlckRpZ2l0cyBpcyBsZXNzIHRoYW4gdGhlIGN1cnJlbnQgdmFsdWUKICAgICAqIG9mIG1pbmltdW1JbnRlZ2VyRGlnaXRzLCB0aGVuIG1pbmltdW1JbnRlZ2VyRGlnaXRzIHdpbGwgYWxzbyBiZSBzZXQgdG8KICAgICAqIHRoZSBuZXcgdmFsdWUuCiAgICAgKgogICAgICogQHNlZSBnZXRNYXhpbXVtSW50ZWdlckRpZ2l0cwogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgc2V0TWF4aW11bUludGVnZXJEaWdpdHMoaW50MzJfdCBuZXdWYWx1ZSk7CgogICAgLyoqCiAgICAgKiBSZXR1cm5zIHRoZSBtaW5pbXVtIG51bWJlciBvZiBkaWdpdHMgYWxsb3dlZCBpbiB0aGUgaW50ZWdlciBwb3J0aW9uIG9mIGEKICAgICAqIG51bWJlci4KICAgICAqIEBzZWUgc2V0TWluaW11bUludGVnZXJEaWdpdHMKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgaW50MzJfdCBnZXRNaW5pbXVtSW50ZWdlckRpZ2l0cyh2b2lkKSBjb25zdDsKCiAgICAvKioKICAgICAqIFNldHMgdGhlIG1pbmltdW0gbnVtYmVyIG9mIGRpZ2l0cyBhbGxvd2VkIGluIHRoZSBpbnRlZ2VyIHBvcnRpb24gb2YgYQogICAgICogbnVtYmVyLiBtaW5pbXVtSW50ZWdlckRpZ2l0cyBtdXN0IGJlICZsdDs9IG1heGltdW1JbnRlZ2VyRGlnaXRzLiAgSWYgdGhlCiAgICAgKiBuZXcgdmFsdWUgZm9yIG1pbmltdW1JbnRlZ2VyRGlnaXRzIGV4Y2VlZHMgdGhlIGN1cnJlbnQgdmFsdWUKICAgICAqIG9mIG1heGltdW1JbnRlZ2VyRGlnaXRzLCB0aGVuIG1heGltdW1JbnRlZ2VyRGlnaXRzIHdpbGwgYWxzbyBiZSBzZXQgdG8KICAgICAqIHRoZSBuZXcgdmFsdWUuCiAgICAgKiBAc2VlIGdldE1pbmltdW1JbnRlZ2VyRGlnaXRzCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBzZXRNaW5pbXVtSW50ZWdlckRpZ2l0cyhpbnQzMl90IG5ld1ZhbHVlKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgdGhlIG1heGltdW0gbnVtYmVyIG9mIGRpZ2l0cyBhbGxvd2VkIGluIHRoZSBmcmFjdGlvbiBwb3J0aW9uIG9mIGEKICAgICAqIG51bWJlci4KICAgICAqIEBzZWUgc2V0TWF4aW11bUZyYWN0aW9uRGlnaXRzCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIGludDMyX3QgZ2V0TWF4aW11bUZyYWN0aW9uRGlnaXRzKHZvaWQpIGNvbnN0OwoKICAgIC8qKgogICAgICogU2V0cyB0aGUgbWF4aW11bSBudW1iZXIgb2YgZGlnaXRzIGFsbG93ZWQgaW4gdGhlIGZyYWN0aW9uIHBvcnRpb24gb2YgYQogICAgICogbnVtYmVyLiBtYXhpbXVtRnJhY3Rpb25EaWdpdHMgbXVzdCBiZSA+PSBtaW5pbXVtRnJhY3Rpb25EaWdpdHMuICBJZiB0aGUKICAgICAqIG5ldyB2YWx1ZSBmb3IgbWF4aW11bUZyYWN0aW9uRGlnaXRzIGlzIGxlc3MgdGhhbiB0aGUgY3VycmVudCB2YWx1ZQogICAgICogb2YgbWluaW11bUZyYWN0aW9uRGlnaXRzLCB0aGVuIG1pbmltdW1GcmFjdGlvbkRpZ2l0cyB3aWxsIGFsc28gYmUgc2V0IHRvCiAgICAgKiB0aGUgbmV3IHZhbHVlLgogICAgICogQHNlZSBnZXRNYXhpbXVtRnJhY3Rpb25EaWdpdHMKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldE1heGltdW1GcmFjdGlvbkRpZ2l0cyhpbnQzMl90IG5ld1ZhbHVlKTsKCiAgICAvKioKICAgICAqIFJldHVybnMgdGhlIG1pbmltdW0gbnVtYmVyIG9mIGRpZ2l0cyBhbGxvd2VkIGluIHRoZSBmcmFjdGlvbiBwb3J0aW9uIG9mIGEKICAgICAqIG51bWJlci4KICAgICAqIEBzZWUgc2V0TWluaW11bUZyYWN0aW9uRGlnaXRzCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIGludDMyX3QgZ2V0TWluaW11bUZyYWN0aW9uRGlnaXRzKHZvaWQpIGNvbnN0OwoKICAgIC8qKgogICAgICogU2V0cyB0aGUgbWluaW11bSBudW1iZXIgb2YgZGlnaXRzIGFsbG93ZWQgaW4gdGhlIGZyYWN0aW9uIHBvcnRpb24gb2YgYQogICAgICogbnVtYmVyLiBtaW5pbXVtRnJhY3Rpb25EaWdpdHMgbXVzdCBiZSAmbHQ7PSBtYXhpbXVtRnJhY3Rpb25EaWdpdHMuICAgSWYgdGhlCiAgICAgKiBuZXcgdmFsdWUgZm9yIG1pbmltdW1GcmFjdGlvbkRpZ2l0cyBleGNlZWRzIHRoZSBjdXJyZW50IHZhbHVlCiAgICAgKiBvZiBtYXhpbXVtRnJhY3Rpb25EaWdpdHMsIHRoZW4gbWF4aW11bUludGVnZXJEaWdpdHMgd2lsbCBhbHNvIGJlIHNldCB0bwogICAgICogdGhlIG5ldyB2YWx1ZQogICAgICogQHNlZSBnZXRNaW5pbXVtRnJhY3Rpb25EaWdpdHMKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldE1pbmltdW1GcmFjdGlvbkRpZ2l0cyhpbnQzMl90IG5ld1ZhbHVlKTsKCnB1YmxpYzoKCiAgICAvKioKICAgICAqIFJldHVybiB0aGUgY2xhc3MgSUQgZm9yIHRoaXMgY2xhc3MuICBUaGlzIGlzIHVzZWZ1bCBvbmx5IGZvcgogICAgICogY29tcGFyaW5nIHRvIGEgcmV0dXJuIHZhbHVlIGZyb20gZ2V0RHluYW1pY0NsYXNzSUQoKS4gIEZvciBleGFtcGxlOgogICAgICogPHByZT4KICAgICAqIC4gICBCYXNlKiBwb2x5bW9ycGhpY19wb2ludGVyID0gY3JlYXRlUG9seW1vcnBoaWNPYmplY3QoKTsKICAgICAqIC4gICBpZiAocG9seW1vcnBoaWNfcG9pbnRlci0+Z2V0RHluYW1pY0NsYXNzSUQoKSA9PQogICAgICogLiAgICAgICBEZXJpdmVkOjpnZXRTdGF0aWNDbGFzc0lEKCkpIC4uLgogICAgICogPC9wcmU+CiAgICAgKiBAcmV0dXJuIFRoZSBjbGFzcyBJRCBmb3IgYWxsIG9iamVjdHMgb2YgdGhpcyBjbGFzcy4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgc3RhdGljIFVDbGFzc0lEIGdldFN0YXRpY0NsYXNzSUQodm9pZCkgeyByZXR1cm4gKFVDbGFzc0lEKSZmZ0NsYXNzSUQ7IH0KCiAgICAvKioKICAgICAqIE92ZXJyaWRlIENhbGVuZGFyCiAgICAgKiBSZXR1cm5zIGEgdW5pcXVlIGNsYXNzIElEIFBPTFlNT1JQSElDQUxMWS4gIFB1cmUgdmlydHVhbCBvdmVycmlkZS4KICAgICAqIFRoaXMgbWV0aG9kIGlzIHRvIGltcGxlbWVudCBhIHNpbXBsZSB2ZXJzaW9uIG9mIFJUVEksIHNpbmNlIG5vdCBhbGwKICAgICAqIEMrKyBjb21waWxlcnMgc3VwcG9ydCBnZW51aW5lIFJUVEkuICBQb2x5bW9ycGhpYyBvcGVyYXRvcj09KCkgYW5kCiAgICAgKiBjbG9uZSgpIG1ldGhvZHMgY2FsbCB0aGlzIG1ldGhvZC4KICAgICAqIDxQPgogICAgICogQHJldHVybiBUaGUgY2xhc3MgSUQgZm9yIHRoaXMgb2JqZWN0LiBBbGwgb2JqZWN0cyBvZiBhCiAgICAgKiBnaXZlbiBjbGFzcyBoYXZlIHRoZSBzYW1lIGNsYXNzIElELiAgT2JqZWN0cyBvZgogICAgICogb3RoZXIgY2xhc3NlcyBoYXZlIGRpZmZlcmVudCBjbGFzcyBJRHMuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgVUNsYXNzSUQgZ2V0RHluYW1pY0NsYXNzSUQodm9pZCkgY29uc3QgeyByZXR1cm4gZ2V0U3RhdGljQ2xhc3NJRCgpOyB9Cgpwcm90ZWN0ZWQ6CgogICAgLyoqCiAgICAgKiBEZWZhdWx0IGNvbnN0cnVjdG9yIGZvciBzdWJjbGFzcyB1c2Ugb25seS4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgTnVtYmVyRm9ybWF0KCk7CgogICAgLyoqCiAgICAgKiBDb3B5IGNvbnN0cnVjdG9yLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBOdW1iZXJGb3JtYXQoY29uc3QgTnVtYmVyRm9ybWF0Jik7CgogICAgLyoqCiAgICAgKiBBc3NpZ25tZW50IG9wZXJhdG9yLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBOdW1iZXJGb3JtYXQmIG9wZXJhdG9yPShjb25zdCBOdW1iZXJGb3JtYXQmKTsKCnByb3RlY3RlZDoKICAgIHN0YXRpYyBjb25zdCBpbnQzMl90IGZnTWF4SW50ZWdlckRpZ2l0czsKICAgIHN0YXRpYyBjb25zdCBpbnQzMl90IGZnTWluSW50ZWdlckRpZ2l0czsKCnByaXZhdGU6CiAgICBzdGF0aWMgY2hhciBmZ0NsYXNzSUQ7CgogICAgZW51bSBFU3R5bGVzIHsKICAgICAgICBrTnVtYmVyU3R5bGUsCiAgICAgICAga0N1cnJlbmN5U3R5bGUsCiAgICAgICAga1BlcmNlbnRTdHlsZSwKICAgICAgICBrU2NpZW50aWZpY1N0eWxlLAogICAgICAgIGtTdHlsZUNvdW50IC8vIEFMV0FZUyBMQVNUIEVOVU06IG51bWJlciBvZiBzdHlsZXMKICAgIH07CgogICAgc3RhdGljIE51bWJlckZvcm1hdCogY3JlYXRlSW5zdGFuY2UoY29uc3QgTG9jYWxlJiBkZXNpcmVkTG9jYWxlLCBFU3R5bGVzIGNob2ljZSwgVUVycm9yQ29kZSYgc3VjY2Vzcyk7CgogICAgc3RhdGljIGNvbnN0IGludDMyX3QgICAgICAgICBmZ051bWJlclBhdHRlcm5zQ291bnQ7CiAgICBzdGF0aWMgY29uc3QgVW5pY29kZVN0cmluZyAgICAgZmdMYXN0UmVzb3J0TnVtYmVyUGF0dGVybnNbXTsKCiAgICBib29sX3QgICAgICBmR3JvdXBpbmdVc2VkOwogICAgaW50MzJfdCAgICAgZk1heEludGVnZXJEaWdpdHM7CiAgICBpbnQzMl90ICAgICBmTWluSW50ZWdlckRpZ2l0czsKICAgIGludDMyX3QgICAgIGZNYXhGcmFjdGlvbkRpZ2l0czsKICAgIGludDMyX3QgICAgIGZNaW5GcmFjdGlvbkRpZ2l0czsKICAgIGJvb2xfdCAgICAgIGZQYXJzZUludGVnZXJPbmx5Owp9OwogCi8vIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KIAppbmxpbmUgYm9vbF90Ck51bWJlckZvcm1hdDo6aXNQYXJzZUludGVnZXJPbmx5KCkgY29uc3QKewogICAgcmV0dXJuIGZQYXJzZUludGVnZXJPbmx5Owp9CgppbmxpbmUgVW5pY29kZVN0cmluZyYKTnVtYmVyRm9ybWF0Ojpmb3JtYXQoY29uc3QgRm9ybWF0dGFibGUmIG9iaiwKICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0IHsKICAgIHJldHVybiBGb3JtYXQ6OmZvcm1hdChvYmosIHJlc3VsdCwgc3RhdHVzKTsKfQogCiNlbmRpZiAvLyBfTlVNRk1UCi8vZW9mCg==