LyoKKiBDb3B5cmlnaHQgqSB7MTk5Ny0xOTk5fSwgSW50ZXJuYXRpb25hbCBCdXNpbmVzcyBNYWNoaW5lcyBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLiBBbGwgUmlnaHRzIFJlc2VydmVkLgoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgoqCiogRmlsZSBNU0dGTVQuSAoqCiogTW9kaWZpY2F0aW9uIEhpc3Rvcnk6CioKKiAgIERhdGUgICAgICAgIE5hbWUgICAgICAgIERlc2NyaXB0aW9uCiogICAwMi8xOS85NyAgICBhbGl1ICAgICAgICBDb252ZXJ0ZWQgZnJvbSBqYXZhLgoqICAgMDMvMjAvOTcgICAgaGVsZW5hICAgICAgRmluaXNoZWQgZmlyc3QgY3V0IG9mIGltcGxlbWVudGF0aW9uLgoqICAgIDA3LzIyLzk4ICAgIHN0ZXBoZW4gICAgICAgIFJlbW92ZWQgb3BlcmF0b3IhPSAoZGVmaW5lZCBpbiBGb3JtYXQpCioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiovCi8vICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi8vIFRoaXMgZmlsZSB3YXMgZ2VuZXJhdGVkIGZyb20gdGhlIGphdmEgc291cmNlIGZpbGUgTWVzc2FnZUZvcm1hdC5qYXZhCi8vICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAKI2lmbmRlZiBNU0dGTVRfSAojZGVmaW5lIE1TR0ZNVF9ICiAKI2luY2x1ZGUgInVuaWNvZGUvdXR5cGVzLmgiCiNpbmNsdWRlICJ1bmljb2RlL2Zvcm1hdC5oIgojaW5jbHVkZSAidW5pY29kZS9sb2NpZC5oIgpjbGFzcyBOdW1iZXJGb3JtYXQ7CgovKioKICogUHJvdmlkZXMgbWVhbnMgdG8gcHJvZHVjZSBjb25jYXRlbmF0ZWQgbWVzc2FnZXMgaW4gbGFuZ3VhZ2UtbmV1dHJhbCB3YXkuCiAqIFVzZSB0aGlzIGZvciBhbGwgY29uY2F0ZW5hdGlvbnMgdGhhdCBzaG93IHVwIHRvIGVuZCB1c2Vycy4KICogPFA+CiAqIFRha2VzIGEgc2V0IG9mIG9iamVjdHMsIGZvcm1hdHMgdGhlbSwgdGhlbiBpbnNlcnRzIHRoZSBmb3JtYXR0ZWQKICogc3RyaW5ncyBpbnRvIHRoZSBwYXR0ZXJuIGF0IHRoZSBhcHByb3ByaWF0ZSBwbGFjZXMuCiAqIDxQPgogKiBIZXJlIGFyZSBzb21lIGV4YW1wbGVzIG9mIHVzYWdlOgogKiBFeGFtcGxlIDE6CiAqIDxwcmU+CiAqIC4gICAgVUVycm9yQ29kZSBzdWNjZXNzID0gVV9aRVJPX0VSUk9SOwogKiAuICAgIEdyZWdvcmlhbkNhbGVuZGFyIGNhbChzdWNjZXNzKTsKICogLiAgICBGb3JtYXR0YWJsZSBhcmd1bWVudHNbXSA9IHsKICogLiAgICAgICAgN0wsCiAqIC4gICAgICAgIEZvcm1hdHRhYmxlKCAoRGF0ZSkgY2FsLmdldFRpbWUoc3VjY2VzcyksIEZvcm1hdHRhYmxlOjprSXNEYXRlKSwKICogLiAgICAgICAgImEgZGlzdHVyYmFuY2UgaW4gdGhlIEZvcmNlIgogKiAuICAgIH07CiAqIC4gICAgCiAqIC4gICAgVW5pY29kZVN0cmluZyByZXN1bHQ7CiAqIC4gICAgTWVzc2FnZUZvcm1hdDo6Zm9ybWF0KAogKiAuICAgICAgICAgIkF0IHsxLHRpbWV9IG9uIHsxLGRhdGV9LCB0aGVyZSB3YXMgezJ9IG9uIHBsYW5ldCB7MCxudW1iZXJ9LiIsCiAqIC4gICAgICAgICBhcmd1bWVudHMsIDMsIHJlc3VsdCwgc3VjY2VzcyApOwogKiAuICAgIAogKiAuICAgIGNvdXQgJmx0OyZsdDsgInJlc3VsdDogIiAmbHQ7Jmx0OyByZXN1bHQgJmx0OyZsdDsgZW5kbDsKICogLiAgICAvLyZsdDtvdXRwdXQ+OiBBdCA0OjM0OjIwIFBNIG9uIDIzLU1hci05OCwgdGhlcmUgd2FzIGEgZGlzdHVyYmFuY2UKICogLiAgICAvLyAgICAgICAgICAgICBpbiB0aGUgRm9yY2Ugb24gcGxhbmV0IDcuCiAqIDwvcHJlPiAgCiAqIFR5cGljYWxseSwgdGhlIG1lc3NhZ2UgZm9ybWF0IHdpbGwgY29tZSBmcm9tIHJlc291cmNlcywgYW5kIHRoZQogKiBhcmd1bWVudHMgd2lsbCBiZSBkeW5hbWljYWxseSBzZXQgYXQgcnVudGltZS4KICogPFA+CiAqIEV4YW1wbGUgMjoKICogPHByZT4KICogLiAgICBzdWNjZXNzID0gVV9aRVJPX0VSUk9SOwogKiAuICAgIEZvcm1hdHRhYmxlIHRlc3RBcmdzW10gPSB7M0wsICJNeURpc2sifTsKICogLiAgIAogKiAuICAgIE1lc3NhZ2VGb3JtYXQqIGZvcm0gPSBuZXcgTWVzc2FnZUZvcm1hdCgKICogLiAgICAgICAgIlRoZSBkaXNrIFwiezF9XCIgY29udGFpbnMgezB9IGZpbGUocykuIiwgc3VjY2VzcyApOwogKiAuICAgICAgICAKICogLiAgICBVbmljb2RlU3RyaW5nIHN0cmluZzsKICogLiAgICBGaWVsZFBvc2l0aW9uIGZwb3MgPSAwOwogKiAuICAgIGNvdXQgJmx0OyZsdDsgImZvcm1hdDogIiAmbHQ7Jmx0OyBmb3JtLT5mb3JtYXQodGVzdEFyZ3MsIDIsIHN0cmluZywgZnBvcywgc3VjY2VzcyApICZsdDsmbHQ7IGVuZGw7CiAqIC4KICogLiAgICAvLyBvdXRwdXQsIHdpdGggZGlmZmVyZW50IHRlc3RBcmdzOgogKiAuICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgMCBmaWxlKHMpLgogKiAuICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgMSBmaWxlKHMpLgogKiAuICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgMSwyNzMgZmlsZShzKS4KICogLiAgICBkZSBsZXRlIGZvcm07CiAqICA8L3ByZT4KICoKICogIFRoZSBwYXR0ZXJuIGlzIG9mIHRoZSBmb2xsb3dpbmcgZm9ybS4gIExlZ2VuZDoKICogIDxwcmU+CiAqIC4gICAgICB7b3B0aW9uYWwgaXRlbX0KICogLiAgICAgIChncm91cCB0aGF0IG1heSBiZSByZXBlYXRlZCkqCiAqICA8L3ByZT4KICogIERvIG5vdCBjb25mdXNlIG9wdGlvbmFsIGl0ZW1zIHdpdGggaXRlbXMgaW5zaWRlIHF1b3RlcyBicmFjZXMsIHN1Y2gKICogIGFzIHRoaXM6ICJ7Ii4gIFF1b3RlZCBicmFjZXMgYXJlIGxpdGVyYWxzLgogKiAgPHByZT4KICogLiAgICAgIG1lc3NhZ2VGb3JtYXRQYXR0ZXJuIDo9IHN0cmluZyAoICJ7IiBtZXNzYWdlRm9ybWF0RWxlbWVudCAifSIgc3RyaW5nICkqCiAqIC4gICAgICAgCiAqIC4gICAgICBtZXNzYWdlRm9ybWF0RWxlbWVudCA6PSBhcmd1bWVudCB7ICIsIiBlbGVtZW50Rm9ybWF0IH0KICogLiAgICAgICAKICogLiAgICAgIGVsZW1lbnRGb3JtYXQgOj0gInRpbWUiIHsgIiwiIGRhdGV0aW1lU3R5bGUgfQogKiAuICAgICAgICAgICAgICAgICAgICAgfCAiZGF0ZSIgeyAiLCIgZGF0ZXRpbWVTdHlsZSB9CiAqIC4gICAgICAgICAgICAgICAgICAgICB8ICJudW1iZXIiIHsgIiwiIG51bWJlclN0eWxlIH0KICogLiAgICAgICAgICAgICAgICAgICAgIHwgImNob2ljZSIgIiwiIGNob2ljZVN0eWxlCiAqIC4gIAogKiAuICAgICAgZGF0ZXRpbWVTdHlsZSA6PSAic2hvcnQiCiAqIC4gICAgICAgICAgICAgICAgICAgICB8ICJtZWRpdW0iCiAqIC4gICAgICAgICAgICAgICAgICAgICB8ICJsb25nIgogKiAuICAgICAgICAgICAgICAgICAgICAgfCAiZnVsbCIKICogLiAgICAgICAgICAgICAgICAgICAgIHwgZGF0ZUZvcm1hdFBhdHRlcm4KICogLgogKiAuICAgICAgbnVtYmVyU3R5bGUgOj0gICAiY3VycmVuY3kiCiAqIC4gICAgICAgICAgICAgICAgICAgICB8ICJwZXJjZW50IgogKiAuICAgICAgICAgICAgICAgICAgICAgfCAiaW50ZWdlciIKICogLiAgICAgICAgICAgICAgICAgICAgIHwgbnVtYmVyRm9ybWF0UGF0dGVybgogKiAuIAogKiAuICAgICAgY2hvaWNlU3R5bGUgOj0gICBjaG9pY2VGb3JtYXRQYXR0ZXJuCiAqIDwvcHJlPgogKiBJZiB0aGVyZSBpcyBubyBlbGVtZW50Rm9ybWF0LCB0aGVuIHRoZSBhcmd1bWVudCBtdXN0IGJlIGEgc3RyaW5nLAogKiB3aGljaCBpcyBzdWJzdGl0dXRlZC4gSWYgdGhlcmUgaXMgbm8gZGF0ZVRpbWVTdHlsZSBvciBudW1iZXJTdHlsZSwKICogdGhlbiB0aGUgZGVmYXVsdCBmb3JtYXQgaXMgdXNlZCAoZS5nLiAgTnVtYmVyRm9ybWF0LmdldEluc3RhbmNlKCksCiAqIERhdGVGb3JtYXQuZ2V0RGVmYXVsdFRpbWUoKSBvciBEYXRlRm9ybWF0LmdldERlZmF1bHREYXRlKCkuIEZvcgogKiBhIENob2ljZUZvcm1hdCwgdGhlIHBhdHRlcm4gbXVzdCBhbHdheXMgYmUgc3BlY2lmaWVkLCBzaW5jZSB0aGVyZQogKiBpcyBubyBkZWZhdWx0LgogKiA8UD4KICogSW4gc3RyaW5ncywgc2luZ2xlIHF1b3RlcyBjYW4gYmUgdXNlZCB0byBxdW90ZSB0aGUgInsiIHNpZ24gaWYKICogbmVjZXNzYXJ5LiBBIHJlYWwgc2luZ2xlIHF1b3RlIGlzIHJlcHJlc2VudGVkIGJ5ICcnLiAgSW5zaWRlIGEKICogbWVzc2FnZUZvcm1hdEVsZW1lbnQsIHF1b3RlcyBhcmUgW25vdF0gcmVtb3ZlZC4gRm9yIGV4YW1wbGUsCiAqIHsxLG51bWJlciwkJyMnLCMjfSB3aWxsIHByb2R1Y2UgYSBudW1iZXIgZm9ybWF0IHdpdGggdGhlIHBvdW5kLXNpZ24KICogcXVvdGVkLCB3aXRoIGEgcmVzdWx0IHN1Y2ggYXM6ICIkIzMxLDQ1Ii4KICogPFA+CiAqIElmIGEgcGF0dGVybiBpcyB1c2VkLCB0aGVuIHVucXVvdGVkIGJyYWNlcyBpbiB0aGUgcGF0dGVybiwgaWYgYW55LAogKiBtdXN0IG1hdGNoOiB0aGF0IGlzLCAiYWIgezB9IGRlIiBhbmQgImFiICd9JyBkZSIgYXJlIG9rLCBidXQgImFiCiAqIHswJ30nIGRlIiBhbmQgImFiIH0gZGUiIGFyZSBub3QuCiAqIDxQPgogKiBUaGUgYXJndW1lbnQgaXMgYSBudW1iZXIgZnJvbSAwIHRvIDksIHdoaWNoIGNvcnJlc3BvbmRzIHRvIHRoZQogKiBhcmd1bWVudHMgcHJlc2VudGVkIGluIGFuIGFycmF5IHRvIGJlIGZvcm1hdHRlZC4KICogPFA+CiAqIEl0IGlzIG9rIHRvIGhhdmUgdW51c2VkIGFyZ3VtZW50cyBpbiB0aGUgYXJyYXkuICBXaXRoIG1pc3NpbmcKICogYXJndW1lbnRzIG9yIGFyZ3VtZW50cyB0aGF0IGFyZSBub3Qgb2YgdGhlIHJpZ2h0IGNsYXNzIGZvciB0aGUKICogc3BlY2lmaWVkIGZvcm1hdCwgYSBmYWlsaW5nIFVFcnJvckNvZGUgcmVzdWx0IGlzIHNldC4KICogPFA+CiAqIEZvciBtb3JlIHNvcGhpc3RpY2F0ZWQgcGF0dGVybnMsIHlvdSBjYW4gdXNlIGEgQ2hvaWNlRm9ybWF0IHRvIGdldAogKiBvdXRwdXQgc3VjaCBhczoKICogPHByZT4KICogLiAgICBVRXJyb3JDb2RlIHN1Y2Nlc3MgPSBVX1pFUk9fRVJST1I7CiAqIC4gICAgTWVzc2FnZUZvcm1hdCogZm9ybSA9IG5ldyBNZXNzYWdlRm9ybWF0KCJUaGUgZGlzayBcInsxfVwiIGNvbnRhaW5zIHswfS4iLCBzdWNjZXNzKTsKICogLiAgICBkb3VibGUgZmlsZWxpbWl0c1tdID0gezAsMSwyfTsKICogLiAgICBVbmljb2RlU3RyaW5nIGZpbGVwYXJ0W10gPSB7Im5vIGZpbGVzIiwib25lIGZpbGUiLCJ7MCxudW1iZXJ9IGZpbGVzIn07CiAqIC4gICAgQ2hvaWNlRm9ybWF0KiBmaWxlZm9ybSA9IG5ldyBDaG9pY2VGb3JtYXQoZmlsZWxpbWl0cywgZmlsZXBhcnQsIDMpOwogKiAuICAgIGZvcm0tPnNldEZvcm1hdCgxLCAqZmlsZWZvcm0pOyAvLyBOT1QgemVybywgc2VlIGJlbG93CiAqIC4gICAgCiAqIC4gICAgRm9ybWF0dGFibGUgdGVzdEFyZ3NbXSA9IHsxMjczTCwgIk15RGlzayJ9OwogKiAuICAgICAKICogLiAgICBVbmljb2RlU3RyaW5nIHN0cmluZzsKICogLiAgICBGaWVsZFBvc2l0aW9uIGZwb3MgPSAwOwogKiAuICAgIGNvdXQgJmx0OyZsdDsgZm9ybS0+Zm9ybWF0KHRlc3RBcmdzLCAyLCBzdHJpbmcsIGZwb3MsIHN1Y2Nlc3MpICZsdDsmbHQ7IGVuZGw7CiAqIC4gICAgCiAqIC4gICAgLy8gb3V0cHV0LCB3aXRoIGRpZmZlcmVudCB0ZXN0QXJncwogKiAuICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgbm8gZmlsZXMuCiAqIC4gICAgLy8gb3V0cHV0OiBUaGUgZGlzayAiTXlEaXNrIiBjb250YWlucyBvbmUgZmlsZS4KICogLiAgICAvLyBvdXRwdXQ6IFRoZSBkaXNrICJNeURpc2siIGNvbnRhaW5zIDEsMjczIGZpbGVzLgogKiA8L3ByZT4KICogWW91IGNhbiBlaXRoZXIgZG8gdGhpcyBwcm9ncmFtbWF0aWNhbGx5LCBhcyBpbiB0aGUgYWJvdmUgZXhhbXBsZSwKICogb3IgYnkgdXNpbmcgYSBwYXR0ZXJuIChzZWUgQ2hvaWNlRm9ybWF0IGZvciBtb3JlIGluZm9ybWF0aW9uKSBhcyBpbjoKICogPHByZT4KICogLiAgIGZvcm0tPmFwcGx5UGF0dGVybigKICogLiAgICAgIlRoZXJlIHswLGNob2ljZSwwI2FyZSBubyBmaWxlc3wxI2lzIG9uZSBmaWxlfDEmbHQ7YXJlIHswLG51bWJlcixpbnRlZ2VyfSBmaWxlc30uIik7CiAqIDwvcHJlPgogKiA8UD4KICogW05vdGU6XSBBcyB3ZSBzZWUgYWJvdmUsIHRoZSBzdHJpbmcgcHJvZHVjZWQgYnkgYSBDaG9pY2VGb3JtYXQgaW4KICogTWVzc2FnZUZvcm1hdCBpcyB0cmVhdGVkIHNwZWNpYWxseTsgb2NjdXJhbmNlcyBvZiAneycgYXJlIHVzZWQgdG8KICogaW5kaWNhdGVkIHN1YmZvcm1hdHMsIGFuZCBjYXVzZSByZWN1cnNpb24uICBJZiB5b3UgY3JlYXRlIGJvdGggYQogKiBNZXNzYWdlRm9ybWF0IGFuZCBDaG9pY2VGb3JtYXQgcHJvZ3JhbW1hdGljYWxseSAoaW5zdGVhZCBvZiB1c2luZwogKiB0aGUgc3RyaW5nIHBhdHRlcm5zKSwgdGhlbiBiZSBjYXJlZnVsIG5vdCB0byBwcm9kdWNlIGEgZm9ybWF0IHRoYXQKICogcmVjdXJzZXMgb24gaXRzZWxmLCB3aGljaCB3aWxsIGNhdXNlIGFuIGluZmluaXRlIGxvb3AuCiAqIDxQPgogKiBbTm90ZTpdIEZvcm1hdHMgYXJlIG51bWJlcmVkIGJ5IG9yZGVyIG9mIHZhcmlhYmxlIGluIHRoZSBzdHJpbmcuCiAqIFRoaXMgaXMgW25vdF0gdGhlIHNhbWUgYXMgdGhlIGFyZ3VtZW50IG51bWJlcmluZyEKICogPHByZT4KICogLiAgIEZvciBleGFtcGxlOiB3aXRoICJhYmN7Mn1kZWZ7M31naGl7MH0uLi4iLAogKiAuICAgCiAqIC4gICBmb3JtYXQwIGFmZmVjdHMgdGhlIGZpcnN0IHZhcmlhYmxlIHsyfQogKiAuICAgZm9ybWF0MSBhZmZlY3RzIHRoZSBzZWNvbmQgdmFyaWFibGUgezN9CiAqIC4gICBmb3JtYXQyIGFmZmVjdHMgdGhlIHNlY29uZCB2YXJpYWJsZSB7MH0KICogPC9wcmU+CiAqIGFuZCBzbyBvbi4KICovCmNsYXNzIFVfSTE4Tl9BUEkgTWVzc2FnZUZvcm1hdCA6IHB1YmxpYyBGb3JtYXQgewpwdWJsaWM6CiAgICBlbnVtIEVGb3JtYXROdW1iZXIgeyBrTWF4Rm9ybWF0ID0gMTAgfTsKICAgIC8qKgogICAgICogQ29uc3RydWN0IGEgbmV3IE1lc3NhZ2VGb3JtYXQgdXNpbmcgdGhlIGdpdmVuIHBhdHRlcm4uCiAgICAgKgogICAgICogQHBhcmFtIHBhdHRlcm4gICBQYXR0ZXJuIHVzZWQgdG8gY29uc3RydWN0IG9iamVjdC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgT3V0cHV0IHBhcmFtIHRvIHJlY2VpdmUgc3VjY2VzcyBjb2RlLiAgSWYgdGhlCiAgICAgKiAgICAgICAgICAgICAgICAgIHBhdHRlcm4gY2Fubm90IGJlIHBhcnNlZCwgc2V0IHRvIGZhaWx1cmUgY29kZS4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgTWVzc2FnZUZvcm1hdChjb25zdCBVbmljb2RlU3RyaW5nJiBwYXR0ZXJuLAogICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICZzdGF0dXMpOwoKICAgIC8qKgogICAgICogQ29uc3RydWN0b3IgdGhhdCBhbGxvd3MgbG9jYWxlIHNwZWNpZmljYXRpb24uCiAgICAgKiBAcGFyYW0gcGF0dGVybiAgIFBhdHRlcm4gdXNlZCB0byBjb25zdHJ1Y3Qgb2JqZWN0LgogICAgICogQHBhcmFtIG5ld0xvY2FsZSBUaGUgbG9jYWxlIHRvIHVzZSBmb3IgZm9ybWF0dGluZyBkYXRlcyBhbmQgbnVtYmVycy4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgT3V0cHV0IHBhcmFtIHRvIHJlY2VpdmUgc3VjY2VzcyBjb2RlLiAgSWYgdGhlCiAgICAgKiAgICAgICAgICAgICAgICAgIHBhdHRlcm4gY2Fubm90IGJlIHBhcnNlZCwgc2V0IHRvIGZhaWx1cmUgY29kZS4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgTWVzc2FnZUZvcm1hdChjb25zdCBVbmljb2RlU3RyaW5nJiBwYXR0ZXJuLAogICAgICAgICAgICAgICAgICBjb25zdCBMb2NhbGUmIG5ld0xvY2FsZSwKICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3VjY2Vzcyk7CgogICAgLyoqCiAgICAgKiBDb3B5IGNvbnN0cnVjdG9yLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBNZXNzYWdlRm9ybWF0KGNvbnN0IE1lc3NhZ2VGb3JtYXQmKTsKCiAgICAvKioKICAgICAqIEFzc2lnbm1lbnQgb3BlcmF0b3IuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIGNvbnN0IE1lc3NhZ2VGb3JtYXQmIG9wZXJhdG9yPShjb25zdCBNZXNzYWdlRm9ybWF0Jik7CgogICAgLyoqCiAgICAgKiBEZXN0cnVjdG9yLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIH5NZXNzYWdlRm9ybWF0KCk7CgogICAgLyoqCiAgICAgKiBDbG9uZSB0aGlzIEZvcm1hdCBvYmplY3QgcG9seW1vcnBoaWNhbGx5LiBUaGUgY2FsbGVyIG93bnMgdGhlCiAgICAgKiByZXN1bHQgYW5kIHNob3VsZCBkZWxldGUgaXQgd2hlbiBkb25lLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIEZvcm1hdCogY2xvbmUodm9pZCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBSZXR1cm4gdHJ1ZSBpZiB0aGUgZ2l2ZW4gRm9ybWF0IG9iamVjdHMgYXJlIHNlbWFudGljYWxseSBlcXVhbC4KICAgICAqIE9iamVjdHMgb2YgZGlmZmVyZW50IHN1YmNsYXNzZXMgYXJlIGNvbnNpZGVyZWQgdW5lcXVhbC4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBVQm9vbCBvcGVyYXRvcj09KGNvbnN0IEZvcm1hdCYgb3RoZXIpIGNvbnN0OwoKICAgIC8qKgogICAgICogU2V0cyB0aGUgbG9jYWxlLiBUaGlzIGxvY2FsZSBpcyB1c2VkIGZvciBmZXRjaGluZyBkZWZhdWx0IG51bWJlciBvciBkYXRlCiAgICAgKiBmb3JtYXQgaW5mb3JtYXRpb24uCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBzZXRMb2NhbGUoY29uc3QgTG9jYWxlJiB0aGVMb2NhbGUpOwoKICAgIC8qKgogICAgICogR2V0cyB0aGUgbG9jYWxlLiBUaGlzIGxvY2FsZSBpcyB1c2VkIGZvciBmZXRjaGluZyBkZWZhdWx0IG51bWJlciBvciBkYXRlCiAgICAgKiBmb3JtYXQgaW5mb3JtYXRpb24uCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgY29uc3QgTG9jYWxlJiBnZXRMb2NhbGUodm9pZCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBBcHBseSB0aGUgZ2l2ZW4gcGF0dGVybiBzdHJpbmcgdG8gdGhpcyBtZXNzYWdlIGZvcm1hdC4KICAgICAqCiAgICAgKiBAcGFyYW0gcGF0dGVybiAgIFRoZSBwYXR0ZXJuIHRvIGJlIGFwcGxpZWQuCiAgICAgKiBAcGFyYW0gc3RhdHVzICAgIE91dHB1dCBwYXJhbSBzZXQgdG8gc3VjY2Vzcy9mYWlsdXJlIGNvZGUgb24KICAgICAqICAgICAgICAgICAgICAgICAgZXhpdC4gSWYgdGhlIHBhdHRlcm4gaXMgaW52YWxpZCwgdGhpcyB3aWxsIGJlCiAgICAgKiAgICAgICAgICAgICAgICAgIHNldCB0byBhIGZhaWx1cmUgcmVzdWx0LgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgYXBwbHlQYXR0ZXJuKGNvbnN0IFVuaWNvZGVTdHJpbmcmIHBhdHRlcm4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cyk7CgogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBwYXR0ZXJuLiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIFVuaWNvZGVTdHJpbmcmIHRvUGF0dGVybihVbmljb2RlU3RyaW5nJiByZXN1bHQpIGNvbnN0OwoKICAgIC8qKgogICAgICogU2V0cyBmb3JtYXRzIHRvIHVzZSBvbiBwYXJhbWV0ZXJzLgogICAgICogU2VlIHRoZSBjbGFzcyBkZXNjcmlwdGlvbiBhYm91dCBmb3JtYXQgbnVtYmVyaW5nLgogICAgICogVGhlIGNhbGxlciBzaG91bGQgbm90IGRlbGV0ZSB0aGUgRm9ybWF0IG9iamVjdHMgYWZ0ZXIgdGhpcyBjYWxsLgogICAgICogQGRyYWZ0IEhTWVM6IHBvc3NpYmxlIHNlbWFudGljIGNoYW5nZSBvbiBsaW1pdGF0aW9uIG9mIHRoZSBzaXplIG9mIGFycmF5CiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBhZG9wdEZvcm1hdHMoRm9ybWF0KiogZm9ybWF0c1RvQWRvcHQsIGludDMyX3QgY291bnQpOwoKICAgIC8qKgogICAgICogU2V0cyBmb3JtYXRzIHRvIHVzZSBvbiBwYXJhbWV0ZXJzLgogICAgICogU2VlIHRoZSBjbGFzcyBkZXNjcmlwdGlvbiBhYm91dCBmb3JtYXQgbnVtYmVyaW5nLgogICAgICogQGRyYWZ0IEhTWVM6IHBvc3NpYmxlIHNlbWFudGljIGNoYW5nZSBvbiBsaW1pdGF0aW9uIG9mIHRoZSBzaXplIG9mIGFycmF5CiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBzZXRGb3JtYXRzKGNvbnN0IEZvcm1hdCoqIG5ld0Zvcm1hdHMsaW50MzJfdCBjbnQpOwoKCiAgICAvKioKICAgICAqIFNldHMgZm9ybWF0cyBpbmRpdmlkdWFsbHkgdG8gdXNlIG9uIHBhcmFtZXRlcnMuCiAgICAgKiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uIGFib3V0IGZvcm1hdCBudW1iZXJpbmcuCiAgICAgKiBUaGUgY2FsbGVyIHNob3VsZCBub3QgZGVsZXRlIHRoZSBGb3JtYXQgb2JqZWN0IGFmdGVyIHRoaXMgY2FsbC4KICAgICAqIEBkcmFmdCBIU1lTOiBwb3NzaWJsZSBzZW1hbnRpYyBjaGFuZ2Ugb24gbGltaXRhdGlvbiBvZiB0aGUgc2l6ZSBvZiBhcnJheQogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgYWRvcHRGb3JtYXQoaW50MzJfdCBmb3JtYXROdW1iZXIsIEZvcm1hdCogZm9ybWF0VG9BZG9wdCk7CgogICAgLyoqCiAgICAgKiBTZXRzIGZvcm1hdHMgaW5kaXZpZHVhbGx5IHRvIHVzZSBvbiBwYXJhbWV0ZXJzLgogICAgICogU2VlIHRoZSBjbGFzcyBkZXNjcmlwdGlvbiBhYm91dCBmb3JtYXQgbnVtYmVyaW5nLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgc2V0Rm9ybWF0KGludDMyX3QgdmFyaWFibGUsIGNvbnN0IEZvcm1hdCYgbmV3Rm9ybWF0KTsKCgogICAgLyoqCiAgICAgKiBHZXRzIGZvcm1hdHMgdGhhdCB3ZXJlIHNldCB3aXRoIHNldEZvcm1hdHMuCiAgICAgKiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uIGFib3V0IGZvcm1hdCBudW1iZXJpbmcuCiAgICAgKiBAZHJhZnQgSFNZUzogcG9zc2libGUgc2VtYW50aWMgY2hhbmdlIG9uIGxpbWl0YXRpb24gb2YgdGhlIHNpemUgb2YgYXJyYXkKICAgICAqLwogICAgdmlydHVhbCBjb25zdCBGb3JtYXQqKiBnZXRGb3JtYXRzKGludDMyX3QmIGNvdW50KSBjb25zdDsKCiAgICAvKioKICAgICAqIFJldHVybnMgcGF0dGVybiB3aXRoIGZvcm1hdHRlZCBvYmplY3RzLiAgRG9lcyBub3QgdGFrZSBvd25lcnNoaXAKICAgICAqIG9mIHRoZSBGb3JtYXR0YWJsZSogYXJyYXk7IGp1c3QgcmVhZHMgaXQgYW5kIHVzZXMgaXQgdG8gZ2VuZXJhdGUKICAgICAqIHRoZSBmb3JtYXQgc3RyaW5nLgogICAgICoKICAgICAqIEBwYXJhbSBzb3VyY2UgICAgQW4gYXJyYXkgb2Ygb2JqZWN0cyB0byBiZSBmb3JtYXR0ZWQgJiBzdWJzdGl0dXRlZC4KICAgICAqIEBwYXJhbSByZXN1bHQgICAgV2hlcmUgdGV4dCBpcyBhcHBlbmRlZC4KICAgICAqIEBwYXJhbSBpZ25vcmUgICAgTm8gdXNlZnVsIHN0YXR1cyBpcyByZXR1cm5lZC4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgVW5pY29kZVN0cmluZyYgZm9ybWF0KCAgY29uc3QgRm9ybWF0dGFibGUqIHNvdXJjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgY291bnQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiByZXN1bHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBGaWVsZFBvc2l0aW9uJiBpZ25vcmUsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdWNjZXNzKSBjb25zdDsKCiAgICAvKioKICAgICAqIENvbnZlbmllbmNlIHJvdXRpbmUuICBBdm9pZHMgZXhwbGljaXQgY3JlYXRpb24gb2YKICAgICAqIE1lc3NhZ2VGb3JtYXQsIGJ1dCBkb2Vzbid0IGFsbG93IGZ1dHVyZSBvcHRpbWl6YXRpb25zLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBzdGF0aWMgVW5pY29kZVN0cmluZyYgZm9ybWF0KCAgIGNvbnN0IFVuaWNvZGVTdHJpbmcmIHBhdHRlcm4sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IEZvcm1hdHRhYmxlKiBhcmd1bWVudHMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgY291bnQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIHJlc3VsdCwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN1Y2Nlc3MpOwoKICAgIC8qKgogICAgICogRm9ybWF0IGFuIG9iamVjdCB0byBwcm9kdWNlIGEgbWVzc2FnZS4gIFRoaXMgbWV0aG9kIGhhbmRsZXMKICAgICAqIEZvcm1hdHRhYmxlIG9iamVjdHMgb2YgdHlwZSBrQXJyYXkuIElmIHRoZSBGb3JtYXR0YWJsZQogICAgICogb2JqZWN0IHR5cGUgaXMgbm90IG9mIHR5cGUga0FycmF5LCB0aGVuIGl0IHJldHVybnMgYSBmYWlsaW5nCiAgICAgKiBVRXJyb3JDb2RlLgogICAgICoKICAgICAqIEBwYXJhbSBvYmogICAgICAgICAgIFRoZSBvYmplY3QgdG8gZm9ybWF0CiAgICAgKiBAcGFyYW0gdG9BcHBlbmRUbyAgICBXaGVyZSB0aGUgdGV4dCBpcyB0byBiZSBhcHBlbmRlZAogICAgICogQHBhcmFtIHBvcyAgICAgICAgICAgT24gaW5wdXQ6IGFuIGFsaWdubWVudCBmaWVsZCwgaWYgZGVzaXJlZC4KICAgICAqICAgICAgICAgICAgICAgICAgICAgIE9uIG91dHB1dDogdGhlIG9mZnNldHMgb2YgdGhlIGFsaWdubWVudCBmaWVsZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgICAgIE91dHB1dCBwYXJhbSBmaWxsZWQgd2l0aCBzdWNjZXNzL2ZhaWx1cmUgc3RhdHVzLgogICAgICogQHJldHVybiAgICAgICAgICAgICAgVGhlIHZhbHVlIHBhc3NlZCBpbiBhcyB0b0FwcGVuZFRvICh0aGlzIGFsbG93cyBjaGFpbmluZywKICAgICAqICAgICAgICAgICAgICAgICAgICAgIGFzIHdpdGggVW5pY29kZVN0cmluZzo6YXBwZW5kKCkpCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgVW5pY29kZVN0cmluZyYgZm9ybWF0KGNvbnN0IEZvcm1hdHRhYmxlJiBvYmosCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiB0b0FwcGVuZFRvLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRmllbGRQb3NpdGlvbiYgcG9zLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdDsKCiAgICAvKioKICAgICAqIFJlZGVjbGFyZWQgRm9ybWF0IG1ldGhvZC4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgVW5pY29kZVN0cmluZyYgZm9ybWF0KGNvbnN0IEZvcm1hdHRhYmxlJiBvYmosCiAgICAgICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN0YXR1cykgY29uc3Q7CgogICAgLyoqCiAgICAgKiBQYXJzZXMgdGhlIHN0cmluZy4KICAgICAqIDxQPgogICAgICogQ2F2ZWF0czogVGhlIHBhcnNlIG1heSBmYWlsIGluIGEgbnVtYmVyIG9mIGNpcmN1bXN0YW5jZXMuICBGb3IKICAgICAqIGV4YW1wbGU6CiAgICAgKiA8UD4KICAgICAqIElmIG9uZSBvZiB0aGUgYXJndW1lbnRzIGRvZXMgbm90IG9jY3VyIGluIHRoZSBwYXR0ZXJuLgogICAgICogPFA+CiAgICAgKiBJZiB0aGUgZm9ybWF0IG9mIGFuIGFyZ3VtZW50IGlzIGxvc2VzIGluZm9ybWF0aW9uLCBzdWNoIGFzIHdpdGgKICAgICAqIGEgY2hvaWNlIGZvcm1hdCB3aGVyZSBhIGxhcmdlIG51bWJlciBmb3JtYXRzIHRvICJtYW55Ii4KICAgICAqIDxQPgogICAgICogRG9lcyBub3QgeWV0IGhhbmRsZSByZWN1cnNpb24gKHdoZXJlIHRoZSBzdWJzdGl0dXRlZCBzdHJpbmdzCiAgICAgKiBjb250YWluIHtufSByZWZlcmVuY2VzLikKICAgICAqIDxQPgogICAgICogV2lsbCBub3QgYWx3YXlzIGZpbmQgYSBtYXRjaCAob3IgdGhlIGNvcnJlY3QgbWF0Y2gpIGlmIHNvbWUKICAgICAqIHBhcnQgb2YgdGhlIHBhcnNlIGlzIGFtYmlndW91cy4gIEZvciBleGFtcGxlLCBpZiB0aGUgcGF0dGVybgogICAgICogInsxfSx7Mn0iIGlzIHVzZWQgd2l0aCB0aGUgc3RyaW5nIGFyZ3VtZW50cyB7ImEsYiIsICJjIn0sIGl0CiAgICAgKiB3aWxsIGZvcm1hdCBhcyAiYSxiLGMiLiAgV2hlbiB0aGUgcmVzdWx0IGlzIHBhcnNlZCwgaXQgd2lsbAogICAgICogcmV0dXJuIHsiYSIsICJiLGMifS4KICAgICAqIDxQPgogICAgICogSWYgYSBzaW5nbGUgYXJndW1lbnQgaXMgZm9ybWF0dGVkIHR3aWNlIGluIHRoZSBzdHJpbmcsIHRoZW4gdGhlCiAgICAgKiBsYXRlciBwYXJzZSB3aW5zLgogICAgICoKICAgICAqIEBwYXJhbSBzb3VyY2UgICAgU3RyaW5nIHRvIGJlIHBhcnNlZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgT24gaW5wdXQsIHN0YXJ0aW5nIHBvc2l0aW9uIGZvciBwYXJzZS4gT24gb3V0cHV0LAogICAgICogICAgICAgICAgICAgICAgICBmaW5hbCBwb3NpdGlvbiBhZnRlciBwYXJzZS4KICAgICAqIEBwYXJhbSBjb3VudCAgICAgT3V0cHV0IHBhcmFtIHRvIHJlY2VpdmUgc2l6ZSBvZiByZXR1cm5lZCBhcnJheS4KICAgICAqIEByZXN1bHQgICAgICAgICAgQXJyYXkgb2YgRm9ybWF0dGFibGUgb2JqZWN0cywgd2l0aCBsZW5ndGgKICAgICAqICAgICAgICAgICAgICAgICAgJ2NvdW50Jywgb3duZWQgYnkgdGhlIGNhbGxlci4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBGb3JtYXR0YWJsZSogcGFyc2UoIGNvbnN0IFVuaWNvZGVTdHJpbmcmIHNvdXJjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBQYXJzZVBvc2l0aW9uJiBzdGF0dXMsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCYgY291bnQpIGNvbnN0OwoKICAgIC8qKgogICAgICogUGFyc2VzIHRoZSBzdHJpbmcuIERvZXMgbm90IHlldCBoYW5kbGUgcmVjdXJzaW9uICh3aGVyZQogICAgICogdGhlIHN1YnN0aXR1dGVkIHN0cmluZ3MgY29udGFpbiB7bn0gcmVmZXJlbmNlcy4pCiAgICAgKgogICAgICogQHBhcmFtIHNvdXJjZSAgICBTdHJpbmcgdG8gYmUgcGFyc2VkLgogICAgICogQHBhcmFtIGNvdW50ICAgICBPdXRwdXQgcGFyYW0gdG8gcmVjZWl2ZSBzaXplIG9mIHJldHVybmVkIGFycmF5LgogICAgICogQHBhcmFtIHN0YXR1cyAgICBPdXRwdXQgcGFyYW0gdG8gcmVjZWl2ZSBzdWNjZXNzL2Vycm9yIGNvZGUuCiAgICAgKiBAcmVzdWx0ICAgICAgICAgIEFycmF5IG9mIEZvcm1hdHRhYmxlIG9iamVjdHMsIHdpdGggbGVuZ3RoCiAgICAgKiAgICAgICAgICAgICAgICAgICdjb3VudCcsIG93bmVkIGJ5IHRoZSBjYWxsZXIuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgRm9ybWF0dGFibGUqIHBhcnNlKCBjb25zdCBVbmljb2RlU3RyaW5nJiBzb3VyY2UsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCYgY291bnQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdDsKCiAgICAvKioKICAgICAqIFBhcnNlIGEgc3RyaW5nIHRvIHByb2R1Y2UgYW4gb2JqZWN0LiAgVGhpcyBtZXRob2RzIGhhbmRsZXMKICAgICAqIHBhcnNpbmcgb2YgbWVzc2FnZSBzdHJpbmdzIGludG8gYXJyYXlzIG9mIEZvcm1hdHRhYmxlIG9iamVjdHMuCiAgICAgKiBEb2VzIG5vdCB5ZXQgaGFuZGxlIHJlY3Vyc2lvbiAod2hlcmUgdGhlIHN1YnN0aXR1dGVkIHN0cmluZ3MKICAgICAqIGNvbnRhaW4gJW4gcmVmZXJlbmNlcy4pCiAgICAgKiA8UD4KICAgICAqIEJlZm9yZSBjYWxsaW5nLCBzZXQgcGFyc2VfcG9zLmluZGV4IHRvIHRoZSBvZmZzZXQgeW91IHdhbnQgdG8KICAgICAqIHN0YXJ0IHBhcnNpbmcgYXQgaW4gdGhlIHNvdXJjZS4gQWZ0ZXIgY2FsbGluZywgcGFyc2VfcG9zLmluZGV4CiAgICAgKiBpcyB0aGUgZW5kIG9mIHRoZSB0ZXh0IHlvdSBwYXJzZWQuICBJZiBlcnJvciBvY2N1cnMsIGluZGV4IGlzCiAgICAgKiB1bmNoYW5nZWQuCiAgICAgKiA8UD4KICAgICAqIFdoZW4gcGFyc2luZywgbGVhZGluZyB3aGl0ZXNwYWNlIGlzIGRpc2NhcmRlZCAod2l0aCBzdWNjZXNzZnVsCiAgICAgKiBwYXJzZSksIHdoaWxlIHRyYWlsaW5nIHdoaXRlc3BhY2UgaXMgbGVmdCBhcyBpcy4KICAgICAqIDxQPgogICAgICogU2VlIEZvcm1hdDo6cGFyc2VPYmplY3QoKSBmb3IgbW9yZS4KICAgICAqCiAgICAgKiBAcGFyYW0gc291cmNlICAgIFRoZSBzdHJpbmcgdG8gYmUgcGFyc2VkIGludG8gYW4gb2JqZWN0LgogICAgICogQHBhcmFtIHJlc3VsdCAgICBGb3JtYXR0YWJsZSB0byBiZSBzZXQgdG8gdGhlIHBhcnNlIHJlc3VsdC4KICAgICAqICAgICAgICAgICAgICAgICAgSWYgcGFyc2UgZmFpbHMsIHJldHVybiBjb250ZW50cyBhcmUgdW5kZWZpbmVkLgogICAgICogQHBhcmFtIHBhcnNlX3BvcyBUaGUgcG9zaXRpb24gdG8gc3RhcnQgcGFyc2luZyBhdC4gVXBvbiByZXR1cm4KICAgICAqICAgICAgICAgICAgICAgICAgdGhpcyBwYXJhbSBpcyBzZXQgdG8gdGhlIHBvc2l0aW9uIGFmdGVyIHRoZQogICAgICogICAgICAgICAgICAgICAgICBsYXN0IGNoYXJhY3RlciBzdWNjZXNzZnVsbHkgcGFyc2VkLiBJZiB0aGUKICAgICAqICAgICAgICAgICAgICAgICAgc291cmNlIGlzIG5vdCBwYXJzZWQgc3VjY2Vzc2Z1bGx5LCB0aGlzIHBhcmFtCiAgICAgKiAgICAgICAgICAgICAgICAgIHdpbGwgcmVtYWluIHVuY2hhbmdlZC4KICAgICAqIEByZXR1cm4gICAgICAgICAgQSBuZXdseSBjcmVhdGVkIEZvcm1hdHRhYmxlKiBvYmplY3QsIG9yIE5VTEwKICAgICAqICAgICAgICAgICAgICAgICAgb24gZmFpbHVyZS4gIFRoZSBjYWxsZXIgb3ducyB0aGlzIGFuZCBzaG91bGQKICAgICAqICAgICAgICAgICAgICAgICAgZGVsZXRlIGl0IHdoZW4gZG9uZS4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHBhcnNlT2JqZWN0KGNvbnN0IFVuaWNvZGVTdHJpbmcmIHNvdXJjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGb3JtYXR0YWJsZSYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcnNlUG9zaXRpb24mIHBhcnNlX3BvcykgY29uc3Q7CgpwdWJsaWM6CiAgICAvKioKICAgICAqIFJldHVybnMgYSB1bmlxdWUgY2xhc3MgSUQgUE9MWU1PUlBISUNBTExZLiAgUHVyZSB2aXJ0dWFsIG92ZXJyaWRlLgogICAgICogVGhpcyBtZXRob2QgaXMgdG8gaW1wbGVtZW50IGEgc2ltcGxlIHZlcnNpb24gb2YgUlRUSSwgc2luY2Ugbm90IGFsbAogICAgICogQysrIGNvbXBpbGVycyBzdXBwb3J0IGdlbnVpbmUgUlRUSS4gIFBvbHltb3JwaGljIG9wZXJhdG9yPT0oKSBhbmQKICAgICAqIGNsb25lKCkgbWV0aG9kcyBjYWxsIHRoaXMgbWV0aG9kLgogICAgICoKICAgICAqIEByZXR1cm4gICAgICAgICAgVGhlIGNsYXNzIElEIGZvciB0aGlzIG9iamVjdC4gQWxsIG9iamVjdHMgb2YgYQogICAgICogICAgICAgICAgICAgICAgICBnaXZlbiBjbGFzcyBoYXZlIHRoZSBzYW1lIGNsYXNzIElELiAgT2JqZWN0cyBvZgogICAgICogICAgICAgICAgICAgICAgICBvdGhlciBjbGFzc2VzIGhhdmUgZGlmZmVyZW50IGNsYXNzIElEcy4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBVQ2xhc3NJRCBnZXREeW5hbWljQ2xhc3NJRCh2b2lkKSBjb25zdDsKCiAgICAvKioKICAgICAqIFJldHVybiB0aGUgY2xhc3MgSUQgZm9yIHRoaXMgY2xhc3MuICBUaGlzIGlzIHVzZWZ1bCBvbmx5IGZvcgogICAgICogY29tcGFyaW5nIHRvIGEgcmV0dXJuIHZhbHVlIGZyb20gZ2V0RHluYW1pY0NsYXNzSUQoKS4gIEZvciBleGFtcGxlOgogICAgICogPHByZT4KICAgICAqIC4gICBCYXNlKiBwb2x5bW9ycGhpY19wb2ludGVyID0gY3JlYXRlUG9seW1vcnBoaWNPYmplY3QoKTsKICAgICAqIC4gICBpZiAocG9seW1vcnBoaWNfcG9pbnRlci0+Z2V0RHluYW1pY0NsYXNzSUQoKSA9PQogICAgICogLiAgICAgIERlcml2ZWQ6OmdldFN0YXRpY0NsYXNzSUQoKSkgLi4uCiAgICAgKiA8L3ByZT4KICAgICAqIEByZXR1cm4gICAgICAgICAgVGhlIGNsYXNzIElEIGZvciBhbGwgb2JqZWN0cyBvZiB0aGlzIGNsYXNzLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBzdGF0aWMgVUNsYXNzSUQgZ2V0U3RhdGljQ2xhc3NJRCh2b2lkKSB7IHJldHVybiAoVUNsYXNzSUQpJmZnQ2xhc3NJRDsgfQoKcHJpdmF0ZToKICAgIHN0YXRpYyBjaGFyIGZnQ2xhc3NJRDsKICAgIHN0YXRpYyBOdW1iZXJGb3JtYXQqIGZnTnVtYmVyRm9ybWF0OwoKICAgIC8vIGZnTnVtYmVyRm9ybWF0IGlzIGhlbGQgaW4gYSBjYWNoZSBvZiBvbmUuCgogICAgc3RhdGljIE51bWJlckZvcm1hdCogZ2V0TnVtYmVyRm9ybWF0KFVFcnJvckNvZGUgJnN0YXR1cyk7IC8vIGNhbGwgdGhpcyBmdW5jdGlvbiB0byAnY2hlY2sgb3V0JyBhIG51bWJlcmZvcm1hdCBmcm9tIHRoZSBjYWNoZS4KICAgIHN0YXRpYyB2b2lkICAgICAgICAgIHJlbGVhc2VOdW1iZXJGb3JtYXQoTnVtYmVyRm9ybWF0ICphZG9wdCk7IC8vIGNhbGwgdGhpcyBmdW5jdGlvbiB0byAncmV0dXJuJyB0aGUgbnVtYmVyIGZvcm1hdCB0byB0aGUgY2FjaGUuCgogICAgTG9jYWxlICAgICAgICAgICAgICAgICBmTG9jYWxlOwogICAgVW5pY29kZVN0cmluZyAgICAgICAgIGZQYXR0ZXJuOwogICAgLy8gbGF0ZXIsIGFsbG93IG1vcmUgdGhhbiB0ZW4gaXRlbXMKICAgIEZvcm1hdCAgICAgICAgICAgICAgICAgKmZGb3JtYXRzW2tNYXhGb3JtYXRdOwogICAgaW50MzJfdCAgICAgICAgICAgICAqZk9mZnNldHM7CiAgICBpbnQzMl90ICAgICAgICAgICAgIGZDb3VudDsKICAgIGludDMyX3QgICAgICAgICAgICAgKmZBcmd1bWVudE51bWJlcnM7CiAgICBpbnQzMl90ICAgICAgICAgICAgIGZNYXhPZmZzZXQ7CiAgICAKICAgIC8qKgogICAgICogSW50ZXJuYWwgcm91dGluZSB1c2VkIGJ5IGZvcm1hdC4KICAgICAqIEBwYXJhbSByZWN1cnNpb25Qcm90ZWN0aW9uIEluaXRpYWxseSB6ZXJvLiBCaXRzIDAuLjkgYXJlIHVzZWQgdG8gaW5kaWNhdGUKICAgICAqIHRoYXQgYSBwYXJhbWV0ZXIgaGFzIGFscmVhZHkgYmVlbiBzZWVuLCB0byBhdm9pZCByZWN1cnNpb24uICBDdXJyZW50bHkKICAgICAqIHVudXNlZC4KICAgICAqLwogICAgc3RhdGljIGNvbnN0IGludDMyX3QgICAgICAgICBmZ0xpc3RMZW5ndGg7CiAgICBzdGF0aWMgY29uc3QgVW5pY29kZVN0cmluZyAgICAgZmdUeXBlTGlzdFtdOwogICAgc3RhdGljIGNvbnN0IFVuaWNvZGVTdHJpbmcgICAgIGZnTW9kaWZpZXJMaXN0W107CiAgICBzdGF0aWMgY29uc3QgVW5pY29kZVN0cmluZyAgICAgZmdEYXRlTW9kaWZpZXJMaXN0W107CgogICAgLyoqIAogICAgICogRmluZHMgdGhlIHdvcmQgcywgaW4gdGhlIGtleXdvcmQgbGlzdCBhbmQgcmV0dXJucyB0aGUgbG9jYXRlZCBpbmRleC4KICAgICAqIEBwYXJhbSBzIHRoZSBrZXl3b3JkIHRvIGJlIHNlYXJjaGVkIGZvci4KICAgICAqIEBwYXJhbSBsaXN0IHRoZSBsaXN0IG9mIGtleXdvcmRzIHRvIGJlIHNlYXJjaGVkIHdpdGguCiAgICAgKiBAcmV0dXJuIHRoZSBpbmRleCBvZiB0aGUgbGlzdCB3aGljaCBtYXRjaGVzIHRoZSBrZXl3b3JkIHMuCiAgICAgKi8KICAgIHN0YXRpYyBpbnQzMl90IGZpbmRLZXl3b3JkKCBjb25zdCBVbmljb2RlU3RyaW5nJiBzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IFVuaWNvZGVTdHJpbmcqIGxpc3QpOwoKICAgIC8qKgogICAgICogRm9ybWF0cyB0aGUgYXJyYXkgb2YgYXJndW1lbnRzIGFuZCBjb3BpZXMgdGhlIHJlc3VsdCBpbnRvIHRoZSByZXN1bHQgYnVmZmVyLAogICAgICogdXBkYXRlcyB0aGUgZmllbGQgcG9zaXRpb24uCiAgICAgKiBAcGFyYW0gYXJndW1lbnRzIHRoZSBmb3JtYXR0YWJsZSBvYmplY3RzIGFycmF5LgogICAgICogQHBhcmFtIGNudCB0aGUgYXJyYXkgY291bnQuCiAgICAgKiBAcGFyYW0gc3RhdHVzIGZpZWxkIHBvc2l0aW9uIHN0YXR1cy4KICAgICAqIEBwYXJhbSByZWN1cnNpb25Qcm90ZWN0aW9uIEluaXRpYWxseSB6ZXJvLiBCaXRzIDAuLjkgYXJlIHVzZWQgdG8gaW5kaWNhdGUKICAgICAqIHRoYXQgYSBwYXJhbWV0ZXIgaGFzIGFscmVhZHkgYmVlbiBzZWVuLCB0byBhdm9pZCByZWN1cnNpb24uICBDdXJyZW50bHkKICAgICAqIHVudXNlZC4KICAgICAqIEBwYXJhbSBzdWNjZXNzIHRoZSBlcnJvciBjb2RlIHN0YXR1cy4KICAgICAqLwogICAgVW5pY29kZVN0cmluZyYgIGZvcm1hdCggY29uc3QgRm9ybWF0dGFibGUqIGFyZ3VtZW50cywgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90IGNudCwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiByZXN1bHQsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgRmllbGRQb3NpdGlvbiYgc3RhdHVzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgcmVjdXJzaW9uUHJvdGVjdGlvbiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN1Y2Nlc3MpIGNvbnN0OwoKICAgIC8qKgogICAgICogQ2hlY2tzIHRoZSBzZWdtZW50cyBmb3IgdGhlIGNsb3Nlc3QgbWF0Y2hlZCBmb3JtYXQgaW5zdGFuY2UgYW5kCiAgICAgKiB1cGRhdGVzIHRoZSBmb3JtYXQgYXJyYXkgd2l0aCB0aGUgbmV3IGZvcm1hdCBpbnN0YW5jZS4KICAgICAqIEBwYXJhbSBwb3NpdGlvbiB0aGUgbGFzdCBwcm9jZXNzZWQgb2Zmc2V0IGluIHRoZSBwYXR0ZXJuIAogICAgICogQHBhcmFtIG9mZnNldE51bWJlciB0aGUgb2Zmc2V0IG51bWJlciBvZiB0aGUgbGFzdCBwcm9jZXNzZWQgc2VnbWVudAogICAgICogQHBhcmFtIHNlZ21lbnRzIHRoZSBzdHJpbmcgdGhhdCBjb250YWlucyB0aGUgcGFyc2VkIHBhdHRlcm4gc2VnbWVudHMuCiAgICAgKiBAcGFyYW0gc3VjY2VzcyB0aGUgZXJyb3IgY29kZQogICAgICovCiAgICB2b2lkICAgICAgICAgICAgbWFrZUZvcm1hdCggaW50MzJfdCBwb3NpdGlvbiwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBvZmZzZXROdW1iZXIsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcqIHNlZ21lbnRzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdWNjZXNzKTsKCiAgICAvKioKICAgICAqIENvbnZlbmllbmNlIG1ldGhvZCB0aGF0IG91Z2h0IHRvIGJlIGluIE51bWJlckZvcm1hdAogICAgICovCiAgICBOdW1iZXJGb3JtYXQqIGNyZWF0ZUludGVnZXJGb3JtYXQoY29uc3QgTG9jYWxlJiBsb2NhbGUsIFVFcnJvckNvZGUmIHN0YXR1cykgY29uc3Q7CgogICAgLyoqCiAgICAgKiBDaGVja3MgdGhlIHJhbmdlIG9mIHRoZSBzb3VyY2UgdGV4dCB0byBxdW90ZSB0aGUgc3BlY2lhbAogICAgICogY2hhcmFjdGVycywgeyBhbmQgJyBhbmQgY29weSB0byB0YXJnZXQgYnVmZmVyLgogICAgICogQHBhcmFtIHNvdXJjZQogICAgICogQHBhcmFtIHN0YXJ0IHRoZSB0ZXh0IG9mZnNldCB0byBzdGFydCB0aGUgcHJvY2VzcyBvZiBpbiB0aGUgc291cmNlIHN0cmluZwogICAgICogQHBhcmFtIGVuZCB0aGUgdGV4dCBvZmZzZXQgdG8gZW5kIHRoZSBwcm9jZXNzIG9mIGluIHRoZSBzb3VyY2Ugc3RyaW5nCiAgICAgKiBAcGFyYW0gdGFyZ2V0IHRoZSByZXN1bHQgYnVmZmVyCiAgICAgKi8KICAgIHN0YXRpYyB2b2lkIGNvcHlBbmRGaXhRdW90ZXMoY29uc3QgVW5pY29kZVN0cmluZyYgc291cmNlLCBpbnQzMl90IHN0YXJ0LCBpbnQzMl90IGVuZCwgVW5pY29kZVN0cmluZyYgdGFyZ2V0KTsKCiAgICAvKioKICAgICAqIENvbnZlcnRzIGEgc3RyaW5nIHRvIGFuIGludGVnZXIgdmFsdWUgdXNpbmcgYSBkZWZhdWx0IE51bWJlckZvcm1hdCBvYmplY3QKICAgICAqIHdoaWNoIGlzIHN0YXRpYyAoc2hhcmVkIGJ5IGFsbCBNZXNzYWdlRm9ybWF0IGluc3RhbmNlcykuICBUaGlzIHJlcGxhY2VzCiAgICAgKiBhIGNhbGwgdG8gd3RvaSgpLgogICAgICogQHBhcmFtIHN0cmluZyB0aGUgc291cmNlIHN0cmluZyB0byBjb252ZXJ0IHdpdGgKICAgICAqIEBwYXJhbSBzdGF0dXMgdGhlIGVycm9yIGNvZGUuCiAgICAgKiBAcmV0dXJuIHRoZSBjb252ZXJ0ZWQgbnVtYmVyLgogICAgICovCiAgICBzdGF0aWMgaW50MzJfdCBzdG9pKGNvbnN0IFVuaWNvZGVTdHJpbmcmIHN0cmluZywgVUVycm9yQ29kZSYgc3RhdHVzKTsKCiAgICAvKioKICAgICAqIENvbnZlcnRzIGFuIGludGVnZXIgdmFsdWUgdG8gYSBzdHJpbmcgdXNpbmcgYSBkZWZhdWx0IE51bWJlckZvcm1hdCBvYmplY3QKICAgICAqIHdoaWNoIGlzIHN0YXRpYyAoc2hhcmVkIGJ5IGFsbCBNZXNzYWdlRm9ybWF0IGluc3RhbmNlcykuICBUaGlzIHJlcGxhY2VzCiAgICAgKiBhIGNhbGwgdG8gd3RvaSgpLgogICAgICogQHBhcmFtIGkgdGhlIGludGVnZXIgdG8gZm9ybWF0CiAgICAgKiBAcGFyYW0gc3RyaW5nIHRoZSBkZXN0aW5hdGlvbiBzdHJpbmcKICAgICAqIEByZXR1cm4gYSByZWZlcmVuY2UgdG8gc3RyaW5nLgogICAgICovCiAgICBzdGF0aWMgVW5pY29kZVN0cmluZyYgaXRvcyhpbnQzMl90IGksIFVuaWNvZGVTdHJpbmcmIHN0cmluZyk7Cn07CiAKaW5saW5lIFVDbGFzc0lEIApNZXNzYWdlRm9ybWF0OjpnZXREeW5hbWljQ2xhc3NJRCgpIGNvbnN0CnsgCiAgICByZXR1cm4gTWVzc2FnZUZvcm1hdDo6Z2V0U3RhdGljQ2xhc3NJRCgpOyAKfQoKaW5saW5lIFVuaWNvZGVTdHJpbmcmCk1lc3NhZ2VGb3JtYXQ6OmZvcm1hdChjb25zdCBGb3JtYXR0YWJsZSYgb2JqLAogICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdCB7CiAgICByZXR1cm4gRm9ybWF0Ojpmb3JtYXQob2JqLCByZXN1bHQsIHN0YXR1cyk7Cn0KCiNlbmRpZiAvLyBfTVNHRk1UCi8vZW9mCg==