LyoKKiBDb3B5cmlnaHQgqSB7MTk5Ny0xOTk5fSwgSW50ZXJuYXRpb25hbCBCdXNpbmVzcyBNYWNoaW5lcyBDb3Jwb3JhdGlvbiBhbmQgb3RoZXJzLiBBbGwgUmlnaHRzIFJlc2VydmVkLgoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgoqCiogRmlsZSBNU0dGTVQuSAoqCiogTW9kaWZpY2F0aW9uIEhpc3Rvcnk6CioKKiAgIERhdGUgICAgICAgIE5hbWUgICAgICAgIERlc2NyaXB0aW9uCiogICAwMi8xOS85NyAgICBhbGl1ICAgICAgICBDb252ZXJ0ZWQgZnJvbSBqYXZhLgoqICAgMDMvMjAvOTcgICAgaGVsZW5hICAgICAgRmluaXNoZWQgZmlyc3QgY3V0IG9mIGltcGxlbWVudGF0aW9uLgoqICAgIDA3LzIyLzk4ICAgIHN0ZXBoZW4gICAgICAgIFJlbW92ZWQgb3BlcmF0b3IhPSAoZGVmaW5lZCBpbiBGb3JtYXQpCioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiovCi8vICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCi8vIFRoaXMgZmlsZSB3YXMgZ2VuZXJhdGVkIGZyb20gdGhlIGphdmEgc291cmNlIGZpbGUgTWVzc2FnZUZvcm1hdC5qYXZhCi8vICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiAKI2lmbmRlZiBNU0dGTVRfSAojZGVmaW5lIE1TR0ZNVF9ICiAKI2luY2x1ZGUgInVuaWNvZGUvdXR5cGVzLmgiCiNpbmNsdWRlICJ1bmljb2RlL2Zvcm1hdC5oIgojaW5jbHVkZSAidW5pY29kZS9sb2NpZC5oIgpjbGFzcyBOdW1iZXJGb3JtYXQ7CgovKioKICogUHJvdmlkZXMgbWVhbnMgdG8gcHJvZHVjZSBjb25jYXRlbmF0ZWQgbWVzc2FnZXMgaW4gbGFuZ3VhZ2UtbmV1dHJhbCB3YXkuCiAqIFVzZSB0aGlzIGZvciBhbGwgY29uY2F0ZW5hdGlvbnMgdGhhdCBzaG93IHVwIHRvIGVuZCB1c2Vycy4KICogPFA+CiAqIFRha2VzIGEgc2V0IG9mIG9iamVjdHMsIGZvcm1hdHMgdGhlbSwgdGhlbiBpbnNlcnRzIHRoZSBmb3JtYXR0ZWQKICogc3RyaW5ncyBpbnRvIHRoZSBwYXR0ZXJuIGF0IHRoZSBhcHByb3ByaWF0ZSBwbGFjZXMuCiAqIDxQPgogKiBIZXJlIGFyZSBzb21lIGV4YW1wbGVzIG9mIHVzYWdlOgogKiBFeGFtcGxlIDE6CiAqIDxwcmU+CiAqIFxjb2RlCiAqICAgICBVRXJyb3JDb2RlIHN1Y2Nlc3MgPSBVX1pFUk9fRVJST1I7CiAqICAgICBHcmVnb3JpYW5DYWxlbmRhciBjYWwoc3VjY2Vzcyk7CiAqICAgICBGb3JtYXR0YWJsZSBhcmd1bWVudHNbXSA9IHsKICogICAgICAgICA3TCwKICogICAgICAgICBGb3JtYXR0YWJsZSggKERhdGUpIGNhbC5nZXRUaW1lKHN1Y2Nlc3MpLCBGb3JtYXR0YWJsZTo6a0lzRGF0ZSksCiAqICAgICAgICAgImEgZGlzdHVyYmFuY2UgaW4gdGhlIEZvcmNlIgogKiAgICAgfTsKICogICAgIAogKiAgICAgVW5pY29kZVN0cmluZyByZXN1bHQ7CiAqICAgICBNZXNzYWdlRm9ybWF0Ojpmb3JtYXQoCiAqICAgICAgICAgICJBdCB7MSx0aW1lfSBvbiB7MSxkYXRlfSwgdGhlcmUgd2FzIHsyfSBvbiBwbGFuZXQgezAsbnVtYmVyfS4iLAogKiAgICAgICAgICBhcmd1bWVudHMsIDMsIHJlc3VsdCwgc3VjY2VzcyApOwogKiAgICAgCiAqICAgICBjb3V0ICZsdDsmbHQ7ICJyZXN1bHQ6ICIgJmx0OyZsdDsgcmVzdWx0ICZsdDsmbHQ7IGVuZGw7CiAqICAgICAvLyZsdDtvdXRwdXQ+OiBBdCA0OjM0OjIwIFBNIG9uIDIzLU1hci05OCwgdGhlcmUgd2FzIGEgZGlzdHVyYmFuY2UKICogICAgIC8vICAgICAgICAgICAgIGluIHRoZSBGb3JjZSBvbiBwbGFuZXQgNy4KICogXGVuZGNvZGUKICogPC9wcmU+ICAKICogVHlwaWNhbGx5LCB0aGUgbWVzc2FnZSBmb3JtYXQgd2lsbCBjb21lIGZyb20gcmVzb3VyY2VzLCBhbmQgdGhlCiAqIGFyZ3VtZW50cyB3aWxsIGJlIGR5bmFtaWNhbGx5IHNldCBhdCBydW50aW1lLgogKiA8UD4KICogRXhhbXBsZSAyOgogKiA8cHJlPgogKiAgXGNvZGUKICogICAgIHN1Y2Nlc3MgPSBVX1pFUk9fRVJST1I7CiAqICAgICBGb3JtYXR0YWJsZSB0ZXN0QXJnc1tdID0gezNMLCAiTXlEaXNrIn07CiAqICAgIAogKiAgICAgTWVzc2FnZUZvcm1hdCogZm9ybSA9IG5ldyBNZXNzYWdlRm9ybWF0KAogKiAgICAgICAgICJUaGUgZGlzayBcInsxfVwiIGNvbnRhaW5zIHswfSBmaWxlKHMpLiIsIHN1Y2Nlc3MgKTsKICogICAgICAgICAKICogICAgIFVuaWNvZGVTdHJpbmcgc3RyaW5nOwogKiAgICAgRmllbGRQb3NpdGlvbiBmcG9zID0gMDsKICogICAgIGNvdXQgJmx0OyZsdDsgImZvcm1hdDogIiAmbHQ7Jmx0OyBmb3JtLT5mb3JtYXQodGVzdEFyZ3MsIDIsIHN0cmluZywgZnBvcywgc3VjY2VzcyApICZsdDsmbHQ7IGVuZGw7CiAqIAogKiAgICAgLy8gb3V0cHV0LCB3aXRoIGRpZmZlcmVudCB0ZXN0QXJnczoKICogICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgMCBmaWxlKHMpLgogKiAgICAgLy8gb3V0cHV0OiBUaGUgZGlzayAiTXlEaXNrIiBjb250YWlucyAxIGZpbGUocykuCiAqICAgICAvLyBvdXRwdXQ6IFRoZSBkaXNrICJNeURpc2siIGNvbnRhaW5zIDEsMjczIGZpbGUocykuCiAqICAgICBkZWxldGUgZm9ybTsKICogIFxlbmRjb2RlCiAqICA8L3ByZT4KICoKICogIFRoZSBwYXR0ZXJuIGlzIG9mIHRoZSBmb2xsb3dpbmcgZm9ybS4gIExlZ2VuZDoKICogIDxwcmU+CiAqIFxjb2RlCiAqICAgICAgIHtvcHRpb25hbCBpdGVtfQogKiAgICAgICAoZ3JvdXAgdGhhdCBtYXkgYmUgcmVwZWF0ZWQpKgogKiBcZW5kY29kZQogKiAgPC9wcmU+CiAqICBEbyBub3QgY29uZnVzZSBvcHRpb25hbCBpdGVtcyB3aXRoIGl0ZW1zIGluc2lkZSBxdW90ZXMgYnJhY2VzLCBzdWNoCiAqICBhcyB0aGlzOiAieyIuICBRdW90ZWQgYnJhY2VzIGFyZSBsaXRlcmFscy4KICogIDxwcmU+CiAqICBcY29kZQogKiAgICAgICBtZXNzYWdlRm9ybWF0UGF0dGVybiA6PSBzdHJpbmcgKCAieyIgbWVzc2FnZUZvcm1hdEVsZW1lbnQgIn0iIHN0cmluZyApKgogKiAgICAgICAgCiAqICAgICAgIG1lc3NhZ2VGb3JtYXRFbGVtZW50IDo9IGFyZ3VtZW50IHsgIiwiIGVsZW1lbnRGb3JtYXQgfQogKiAgICAgICAgCiAqICAgICAgIGVsZW1lbnRGb3JtYXQgOj0gInRpbWUiIHsgIiwiIGRhdGV0aW1lU3R5bGUgfQogKiAgICAgICAgICAgICAgICAgICAgICB8ICJkYXRlIiB7ICIsIiBkYXRldGltZVN0eWxlIH0KICogICAgICAgICAgICAgICAgICAgICAgfCAibnVtYmVyIiB7ICIsIiBudW1iZXJTdHlsZSB9CiAqICAgICAgICAgICAgICAgICAgICAgIHwgImNob2ljZSIgIiwiIGNob2ljZVN0eWxlCiAqICAgCiAqICAgICAgIGRhdGV0aW1lU3R5bGUgOj0gInNob3J0IgogKiAgICAgICAgICAgICAgICAgICAgICB8ICJtZWRpdW0iCiAqICAgICAgICAgICAgICAgICAgICAgIHwgImxvbmciCiAqICAgICAgICAgICAgICAgICAgICAgIHwgImZ1bGwiCiAqICAgICAgICAgICAgICAgICAgICAgIHwgZGF0ZUZvcm1hdFBhdHRlcm4KICogCiAqICAgICAgIG51bWJlclN0eWxlIDo9ICAgImN1cnJlbmN5IgogKiAgICAgICAgICAgICAgICAgICAgICB8ICJwZXJjZW50IgogKiAgICAgICAgICAgICAgICAgICAgICB8ICJpbnRlZ2VyIgogKiAgICAgICAgICAgICAgICAgICAgICB8IG51bWJlckZvcm1hdFBhdHRlcm4KICogIAogKiAgICAgICBjaG9pY2VTdHlsZSA6PSAgIGNob2ljZUZvcm1hdFBhdHRlcm4KICogXGVuZGNvZGUKICogPC9wcmU+CiAqIElmIHRoZXJlIGlzIG5vIGVsZW1lbnRGb3JtYXQsIHRoZW4gdGhlIGFyZ3VtZW50IG11c3QgYmUgYSBzdHJpbmcsCiAqIHdoaWNoIGlzIHN1YnN0aXR1dGVkLiBJZiB0aGVyZSBpcyBubyBkYXRlVGltZVN0eWxlIG9yIG51bWJlclN0eWxlLAogKiB0aGVuIHRoZSBkZWZhdWx0IGZvcm1hdCBpcyB1c2VkIChlLmcuICBOdW1iZXJGb3JtYXQuZ2V0SW5zdGFuY2UoKSwKICogRGF0ZUZvcm1hdC5nZXREZWZhdWx0VGltZSgpIG9yIERhdGVGb3JtYXQuZ2V0RGVmYXVsdERhdGUoKS4gRm9yCiAqIGEgQ2hvaWNlRm9ybWF0LCB0aGUgcGF0dGVybiBtdXN0IGFsd2F5cyBiZSBzcGVjaWZpZWQsIHNpbmNlIHRoZXJlCiAqIGlzIG5vIGRlZmF1bHQuCiAqIDxQPgogKiBJbiBzdHJpbmdzLCBzaW5nbGUgcXVvdGVzIGNhbiBiZSB1c2VkIHRvIHF1b3RlIHRoZSAieyIgc2lnbiBpZgogKiBuZWNlc3NhcnkuIEEgcmVhbCBzaW5nbGUgcXVvdGUgaXMgcmVwcmVzZW50ZWQgYnkgJycuICBJbnNpZGUgYQogKiBtZXNzYWdlRm9ybWF0RWxlbWVudCwgcXVvdGVzIGFyZSBbbm90XSByZW1vdmVkLiBGb3IgZXhhbXBsZSwKICogezEsbnVtYmVyLCQnIycsIyN9IHdpbGwgcHJvZHVjZSBhIG51bWJlciBmb3JtYXQgd2l0aCB0aGUgcG91bmQtc2lnbgogKiBxdW90ZWQsIHdpdGggYSByZXN1bHQgc3VjaCBhczogIiQjMzEsNDUiLgogKiA8UD4KICogSWYgYSBwYXR0ZXJuIGlzIHVzZWQsIHRoZW4gdW5xdW90ZWQgYnJhY2VzIGluIHRoZSBwYXR0ZXJuLCBpZiBhbnksCiAqIG11c3QgbWF0Y2g6IHRoYXQgaXMsICJhYiB7MH0gZGUiIGFuZCAiYWIgJ30nIGRlIiBhcmUgb2ssIGJ1dCAiYWIKICogezAnfScgZGUiIGFuZCAiYWIgfSBkZSIgYXJlIG5vdC4KICogPFA+CiAqIFRoZSBhcmd1bWVudCBpcyBhIG51bWJlciBmcm9tIDAgdG8gOSwgd2hpY2ggY29ycmVzcG9uZHMgdG8gdGhlCiAqIGFyZ3VtZW50cyBwcmVzZW50ZWQgaW4gYW4gYXJyYXkgdG8gYmUgZm9ybWF0dGVkLgogKiA8UD4KICogSXQgaXMgb2sgdG8gaGF2ZSB1bnVzZWQgYXJndW1lbnRzIGluIHRoZSBhcnJheS4gIFdpdGggbWlzc2luZwogKiBhcmd1bWVudHMgb3IgYXJndW1lbnRzIHRoYXQgYXJlIG5vdCBvZiB0aGUgcmlnaHQgY2xhc3MgZm9yIHRoZQogKiBzcGVjaWZpZWQgZm9ybWF0LCBhIGZhaWxpbmcgVUVycm9yQ29kZSByZXN1bHQgaXMgc2V0LgogKiA8UD4KICogRm9yIG1vcmUgc29waGlzdGljYXRlZCBwYXR0ZXJucywgeW91IGNhbiB1c2UgYSBDaG9pY2VGb3JtYXQgdG8gZ2V0CiAqIG91dHB1dCBzdWNoIGFzOgogKiA8cHJlPgogKiBcY29kZQogKiAgICAgVUVycm9yQ29kZSBzdWNjZXNzID0gVV9aRVJPX0VSUk9SOwogKiAgICAgTWVzc2FnZUZvcm1hdCogZm9ybSA9IG5ldyBNZXNzYWdlRm9ybWF0KCJUaGUgZGlzayBcInsxfVwiIGNvbnRhaW5zIHswfS4iLCBzdWNjZXNzKTsKICogICAgIGRvdWJsZSBmaWxlbGltaXRzW10gPSB7MCwxLDJ9OwogKiAgICAgVW5pY29kZVN0cmluZyBmaWxlcGFydFtdID0geyJubyBmaWxlcyIsIm9uZSBmaWxlIiwiezAsbnVtYmVyfSBmaWxlcyJ9OwogKiAgICAgQ2hvaWNlRm9ybWF0KiBmaWxlZm9ybSA9IG5ldyBDaG9pY2VGb3JtYXQoZmlsZWxpbWl0cywgZmlsZXBhcnQsIDMpOwogKiAgICAgZm9ybS0+c2V0Rm9ybWF0KDEsICpmaWxlZm9ybSk7IC8vIE5PVCB6ZXJvLCBzZWUgYmVsb3cKICogICAgIAogKiAgICAgRm9ybWF0dGFibGUgdGVzdEFyZ3NbXSA9IHsxMjczTCwgIk15RGlzayJ9OwogKiAgICAgIAogKiAgICAgVW5pY29kZVN0cmluZyBzdHJpbmc7CiAqICAgICBGaWVsZFBvc2l0aW9uIGZwb3MgPSAwOwogKiAgICAgY291dCAmbHQ7Jmx0OyBmb3JtLT5mb3JtYXQodGVzdEFyZ3MsIDIsIHN0cmluZywgZnBvcywgc3VjY2VzcykgJmx0OyZsdDsgZW5kbDsKICogICAgIAogKiAgICAgLy8gb3V0cHV0LCB3aXRoIGRpZmZlcmVudCB0ZXN0QXJncwogKiAgICAgLy8gb3V0cHV0OiBUaGUgZGlzayAiTXlEaXNrIiBjb250YWlucyBubyBmaWxlcy4KICogICAgIC8vIG91dHB1dDogVGhlIGRpc2sgIk15RGlzayIgY29udGFpbnMgb25lIGZpbGUuCiAqICAgICAvLyBvdXRwdXQ6IFRoZSBkaXNrICJNeURpc2siIGNvbnRhaW5zIDEsMjczIGZpbGVzLgogKiBcZW5kY29kZQogKiA8L3ByZT4KICogWW91IGNhbiBlaXRoZXIgZG8gdGhpcyBwcm9ncmFtbWF0aWNhbGx5LCBhcyBpbiB0aGUgYWJvdmUgZXhhbXBsZSwKICogb3IgYnkgdXNpbmcgYSBwYXR0ZXJuIChzZWUgQ2hvaWNlRm9ybWF0IGZvciBtb3JlIGluZm9ybWF0aW9uKSBhcyBpbjoKICogPHByZT4KICogXGNvZGUKICogICAgZm9ybS0+YXBwbHlQYXR0ZXJuKAogKiAgICAgICJUaGVyZSB7MCxjaG9pY2UsMCNhcmUgbm8gZmlsZXN8MSNpcyBvbmUgZmlsZXwxJmx0O2FyZSB7MCxudW1iZXIsaW50ZWdlcn0gZmlsZXN9LiIpOwogKiBcZW5kY29kZQogKiA8L3ByZT4KICogPFA+CiAqIFtOb3RlOl0gQXMgd2Ugc2VlIGFib3ZlLCB0aGUgc3RyaW5nIHByb2R1Y2VkIGJ5IGEgQ2hvaWNlRm9ybWF0IGluCiAqIE1lc3NhZ2VGb3JtYXQgaXMgdHJlYXRlZCBzcGVjaWFsbHk7IG9jY3VyYW5jZXMgb2YgJ3snIGFyZSB1c2VkIHRvCiAqIGluZGljYXRlZCBzdWJmb3JtYXRzLCBhbmQgY2F1c2UgcmVjdXJzaW9uLiAgSWYgeW91IGNyZWF0ZSBib3RoIGEKICogTWVzc2FnZUZvcm1hdCBhbmQgQ2hvaWNlRm9ybWF0IHByb2dyYW1tYXRpY2FsbHkgKGluc3RlYWQgb2YgdXNpbmcKICogdGhlIHN0cmluZyBwYXR0ZXJucyksIHRoZW4gYmUgY2FyZWZ1bCBub3QgdG8gcHJvZHVjZSBhIGZvcm1hdCB0aGF0CiAqIHJlY3Vyc2VzIG9uIGl0c2VsZiwgd2hpY2ggd2lsbCBjYXVzZSBhbiBpbmZpbml0ZSBsb29wLgogKiA8UD4KICogW05vdGU6XSBGb3JtYXRzIGFyZSBudW1iZXJlZCBieSBvcmRlciBvZiB2YXJpYWJsZSBpbiB0aGUgc3RyaW5nLgogKiBUaGlzIGlzIFtub3RdIHRoZSBzYW1lIGFzIHRoZSBhcmd1bWVudCBudW1iZXJpbmchCiAqIDxwcmU+CiAqIFxjb2RlCiAqICAgIEZvciBleGFtcGxlOiB3aXRoICJhYmN7Mn1kZWZ7M31naGl7MH0uLi4iLAogKiAgICAKICogICAgZm9ybWF0MCBhZmZlY3RzIHRoZSBmaXJzdCB2YXJpYWJsZSB7Mn0KICogICAgZm9ybWF0MSBhZmZlY3RzIHRoZSBzZWNvbmQgdmFyaWFibGUgezN9CiAqICAgIGZvcm1hdDIgYWZmZWN0cyB0aGUgc2Vjb25kIHZhcmlhYmxlIHswfQogKiBcZW5kY29kZQogKiA8L3ByZT4KICogYW5kIHNvIG9uLgogKi8KY2xhc3MgVV9JMThOX0FQSSBNZXNzYWdlRm9ybWF0IDogcHVibGljIEZvcm1hdCB7CnB1YmxpYzoKICAgIGVudW0gRUZvcm1hdE51bWJlciB7IGtNYXhGb3JtYXQgPSAxMCB9OwogICAgLyoqCiAgICAgKiBDb25zdHJ1Y3QgYSBuZXcgTWVzc2FnZUZvcm1hdCB1c2luZyB0aGUgZ2l2ZW4gcGF0dGVybi4KICAgICAqCiAgICAgKiBAcGFyYW0gcGF0dGVybiAgIFBhdHRlcm4gdXNlZCB0byBjb25zdHJ1Y3Qgb2JqZWN0LgogICAgICogQHBhcmFtIHN0YXR1cyAgICBPdXRwdXQgcGFyYW0gdG8gcmVjZWl2ZSBzdWNjZXNzIGNvZGUuICBJZiB0aGUKICAgICAqICAgICAgICAgICAgICAgICAgcGF0dGVybiBjYW5ub3QgYmUgcGFyc2VkLCBzZXQgdG8gZmFpbHVyZSBjb2RlLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBNZXNzYWdlRm9ybWF0KGNvbnN0IFVuaWNvZGVTdHJpbmcmIHBhdHRlcm4sCiAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUgJnN0YXR1cyk7CgogICAgLyoqCiAgICAgKiBDb25zdHJ1Y3RvciB0aGF0IGFsbG93cyBsb2NhbGUgc3BlY2lmaWNhdGlvbi4KICAgICAqIEBwYXJhbSBwYXR0ZXJuICAgUGF0dGVybiB1c2VkIHRvIGNvbnN0cnVjdCBvYmplY3QuCiAgICAgKiBAcGFyYW0gbmV3TG9jYWxlIFRoZSBsb2NhbGUgdG8gdXNlIGZvciBmb3JtYXR0aW5nIGRhdGVzIGFuZCBudW1iZXJzLgogICAgICogQHBhcmFtIHN0YXR1cyAgICBPdXRwdXQgcGFyYW0gdG8gcmVjZWl2ZSBzdWNjZXNzIGNvZGUuICBJZiB0aGUKICAgICAqICAgICAgICAgICAgICAgICAgcGF0dGVybiBjYW5ub3QgYmUgcGFyc2VkLCBzZXQgdG8gZmFpbHVyZSBjb2RlLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBNZXNzYWdlRm9ybWF0KGNvbnN0IFVuaWNvZGVTdHJpbmcmIHBhdHRlcm4sCiAgICAgICAgICAgICAgICAgIGNvbnN0IExvY2FsZSYgbmV3TG9jYWxlLAogICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdWNjZXNzKTsKCiAgICAvKioKICAgICAqIENvcHkgY29uc3RydWN0b3IuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIE1lc3NhZ2VGb3JtYXQoY29uc3QgTWVzc2FnZUZvcm1hdCYpOwoKICAgIC8qKgogICAgICogQXNzaWdubWVudCBvcGVyYXRvci4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgY29uc3QgTWVzc2FnZUZvcm1hdCYgb3BlcmF0b3I9KGNvbnN0IE1lc3NhZ2VGb3JtYXQmKTsKCiAgICAvKioKICAgICAqIERlc3RydWN0b3IuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgfk1lc3NhZ2VGb3JtYXQoKTsKCiAgICAvKioKICAgICAqIENsb25lIHRoaXMgRm9ybWF0IG9iamVjdCBwb2x5bW9ycGhpY2FsbHkuIFRoZSBjYWxsZXIgb3ducyB0aGUKICAgICAqIHJlc3VsdCBhbmQgc2hvdWxkIGRlbGV0ZSBpdCB3aGVuIGRvbmUuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgRm9ybWF0KiBjbG9uZSh2b2lkKSBjb25zdDsKCiAgICAvKioKICAgICAqIFJldHVybiB0cnVlIGlmIHRoZSBnaXZlbiBGb3JtYXQgb2JqZWN0cyBhcmUgc2VtYW50aWNhbGx5IGVxdWFsLgogICAgICogT2JqZWN0cyBvZiBkaWZmZXJlbnQgc3ViY2xhc3NlcyBhcmUgY29uc2lkZXJlZCB1bmVxdWFsLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIFVCb29sIG9wZXJhdG9yPT0oY29uc3QgRm9ybWF0JiBvdGhlcikgY29uc3Q7CgogICAgLyoqCiAgICAgKiBTZXRzIHRoZSBsb2NhbGUuIFRoaXMgbG9jYWxlIGlzIHVzZWQgZm9yIGZldGNoaW5nIGRlZmF1bHQgbnVtYmVyIG9yIGRhdGUKICAgICAqIGZvcm1hdCBpbmZvcm1hdGlvbi4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldExvY2FsZShjb25zdCBMb2NhbGUmIHRoZUxvY2FsZSk7CgogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBsb2NhbGUuIFRoaXMgbG9jYWxlIGlzIHVzZWQgZm9yIGZldGNoaW5nIGRlZmF1bHQgbnVtYmVyIG9yIGRhdGUKICAgICAqIGZvcm1hdCBpbmZvcm1hdGlvbi4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBjb25zdCBMb2NhbGUmIGdldExvY2FsZSh2b2lkKSBjb25zdDsKCiAgICAvKioKICAgICAqIEFwcGx5IHRoZSBnaXZlbiBwYXR0ZXJuIHN0cmluZyB0byB0aGlzIG1lc3NhZ2UgZm9ybWF0LgogICAgICoKICAgICAqIEBwYXJhbSBwYXR0ZXJuICAgVGhlIHBhdHRlcm4gdG8gYmUgYXBwbGllZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgICAgT3V0cHV0IHBhcmFtIHNldCB0byBzdWNjZXNzL2ZhaWx1cmUgY29kZSBvbgogICAgICogICAgICAgICAgICAgICAgICBleGl0LiBJZiB0aGUgcGF0dGVybiBpcyBpbnZhbGlkLCB0aGlzIHdpbGwgYmUKICAgICAqICAgICAgICAgICAgICAgICAgc2V0IHRvIGEgZmFpbHVyZSByZXN1bHQuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBhcHBseVBhdHRlcm4oY29uc3QgVW5pY29kZVN0cmluZyYgcGF0dGVybiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKTsKCiAgICAvKioKICAgICAqIEdldHMgdGhlIHBhdHRlcm4uIFNlZSB0aGUgY2xhc3MgZGVzY3JpcHRpb24uCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgVW5pY29kZVN0cmluZyYgdG9QYXR0ZXJuKFVuaWNvZGVTdHJpbmcmIHJlc3VsdCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBTZXRzIGZvcm1hdHMgdG8gdXNlIG9uIHBhcmFtZXRlcnMuCiAgICAgKiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uIGFib3V0IGZvcm1hdCBudW1iZXJpbmcuCiAgICAgKiBUaGUgY2FsbGVyIHNob3VsZCBub3QgZGVsZXRlIHRoZSBGb3JtYXQgb2JqZWN0cyBhZnRlciB0aGlzIGNhbGwuCiAgICAgKiBAZHJhZnQgSFNZUzogcG9zc2libGUgc2VtYW50aWMgY2hhbmdlIG9uIGxpbWl0YXRpb24gb2YgdGhlIHNpemUgb2YgYXJyYXkKICAgICAqLwogICAgdmlydHVhbCB2b2lkIGFkb3B0Rm9ybWF0cyhGb3JtYXQqKiBmb3JtYXRzVG9BZG9wdCwgaW50MzJfdCBjb3VudCk7CgogICAgLyoqCiAgICAgKiBTZXRzIGZvcm1hdHMgdG8gdXNlIG9uIHBhcmFtZXRlcnMuCiAgICAgKiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uIGFib3V0IGZvcm1hdCBudW1iZXJpbmcuCiAgICAgKiBAZHJhZnQgSFNZUzogcG9zc2libGUgc2VtYW50aWMgY2hhbmdlIG9uIGxpbWl0YXRpb24gb2YgdGhlIHNpemUgb2YgYXJyYXkKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldEZvcm1hdHMoY29uc3QgRm9ybWF0KiogbmV3Rm9ybWF0cyxpbnQzMl90IGNudCk7CgoKICAgIC8qKgogICAgICogU2V0cyBmb3JtYXRzIGluZGl2aWR1YWxseSB0byB1c2Ugb24gcGFyYW1ldGVycy4KICAgICAqIFNlZSB0aGUgY2xhc3MgZGVzY3JpcHRpb24gYWJvdXQgZm9ybWF0IG51bWJlcmluZy4KICAgICAqIFRoZSBjYWxsZXIgc2hvdWxkIG5vdCBkZWxldGUgdGhlIEZvcm1hdCBvYmplY3QgYWZ0ZXIgdGhpcyBjYWxsLgogICAgICogQGRyYWZ0IEhTWVM6IHBvc3NpYmxlIHNlbWFudGljIGNoYW5nZSBvbiBsaW1pdGF0aW9uIG9mIHRoZSBzaXplIG9mIGFycmF5CiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBhZG9wdEZvcm1hdChpbnQzMl90IGZvcm1hdE51bWJlciwgRm9ybWF0KiBmb3JtYXRUb0Fkb3B0KTsKCiAgICAvKioKICAgICAqIFNldHMgZm9ybWF0cyBpbmRpdmlkdWFsbHkgdG8gdXNlIG9uIHBhcmFtZXRlcnMuCiAgICAgKiBTZWUgdGhlIGNsYXNzIGRlc2NyaXB0aW9uIGFib3V0IGZvcm1hdCBudW1iZXJpbmcuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBzZXRGb3JtYXQoaW50MzJfdCB2YXJpYWJsZSwgY29uc3QgRm9ybWF0JiBuZXdGb3JtYXQpOwoKCiAgICAvKioKICAgICAqIEdldHMgZm9ybWF0cyB0aGF0IHdlcmUgc2V0IHdpdGggc2V0Rm9ybWF0cy4KICAgICAqIFNlZSB0aGUgY2xhc3MgZGVzY3JpcHRpb24gYWJvdXQgZm9ybWF0IG51bWJlcmluZy4KICAgICAqIEBkcmFmdCBIU1lTOiBwb3NzaWJsZSBzZW1hbnRpYyBjaGFuZ2Ugb24gbGltaXRhdGlvbiBvZiB0aGUgc2l6ZSBvZiBhcnJheQogICAgICovCiAgICB2aXJ0dWFsIGNvbnN0IEZvcm1hdCoqIGdldEZvcm1hdHMoaW50MzJfdCYgY291bnQpIGNvbnN0OwoKICAgIC8qKgogICAgICogUmV0dXJucyBwYXR0ZXJuIHdpdGggZm9ybWF0dGVkIG9iamVjdHMuICBEb2VzIG5vdCB0YWtlIG93bmVyc2hpcAogICAgICogb2YgdGhlIEZvcm1hdHRhYmxlKiBhcnJheTsganVzdCByZWFkcyBpdCBhbmQgdXNlcyBpdCB0byBnZW5lcmF0ZQogICAgICogdGhlIGZvcm1hdCBzdHJpbmcuCiAgICAgKgogICAgICogQHBhcmFtIHNvdXJjZSAgICBBbiBhcnJheSBvZiBvYmplY3RzIHRvIGJlIGZvcm1hdHRlZCAmIHN1YnN0aXR1dGVkLgogICAgICogQHBhcmFtIHJlc3VsdCAgICBXaGVyZSB0ZXh0IGlzIGFwcGVuZGVkLgogICAgICogQHBhcmFtIGlnbm9yZSAgICBObyB1c2VmdWwgc3RhdHVzIGlzIHJldHVybmVkLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBVbmljb2RlU3RyaW5nJiBmb3JtYXQoICBjb25zdCBGb3JtYXR0YWJsZSogc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBjb3VudCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIHJlc3VsdCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZpZWxkUG9zaXRpb24mIGlnbm9yZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVFcnJvckNvZGUmIHN1Y2Nlc3MpIGNvbnN0OwoKICAgIC8qKgogICAgICogQ29udmVuaWVuY2Ugcm91dGluZS4gIEF2b2lkcyBleHBsaWNpdCBjcmVhdGlvbiBvZgogICAgICogTWVzc2FnZUZvcm1hdCwgYnV0IGRvZXNuJ3QgYWxsb3cgZnV0dXJlIG9wdGltaXphdGlvbnMuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBVbmljb2RlU3RyaW5nJiBmb3JtYXQoICAgY29uc3QgVW5pY29kZVN0cmluZyYgcGF0dGVybiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgRm9ybWF0dGFibGUqIGFyZ3VtZW50cywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBjb3VudCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgcmVzdWx0LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3VjY2Vzcyk7CgogICAgLyoqCiAgICAgKiBGb3JtYXQgYW4gb2JqZWN0IHRvIHByb2R1Y2UgYSBtZXNzYWdlLiAgVGhpcyBtZXRob2QgaGFuZGxlcwogICAgICogRm9ybWF0dGFibGUgb2JqZWN0cyBvZiB0eXBlIGtBcnJheS4gSWYgdGhlIEZvcm1hdHRhYmxlCiAgICAgKiBvYmplY3QgdHlwZSBpcyBub3Qgb2YgdHlwZSBrQXJyYXksIHRoZW4gaXQgcmV0dXJucyBhIGZhaWxpbmcKICAgICAqIFVFcnJvckNvZGUuCiAgICAgKgogICAgICogQHBhcmFtIG9iaiAgICAgICAgICAgVGhlIG9iamVjdCB0byBmb3JtYXQKICAgICAqIEBwYXJhbSB0b0FwcGVuZFRvICAgIFdoZXJlIHRoZSB0ZXh0IGlzIHRvIGJlIGFwcGVuZGVkCiAgICAgKiBAcGFyYW0gcG9zICAgICAgICAgICBPbiBpbnB1dDogYW4gYWxpZ25tZW50IGZpZWxkLCBpZiBkZXNpcmVkLgogICAgICogICAgICAgICAgICAgICAgICAgICAgT24gb3V0cHV0OiB0aGUgb2Zmc2V0cyBvZiB0aGUgYWxpZ25tZW50IGZpZWxkLgogICAgICogQHBhcmFtIHN0YXR1cyAgICAgICAgT3V0cHV0IHBhcmFtIGZpbGxlZCB3aXRoIHN1Y2Nlc3MvZmFpbHVyZSBzdGF0dXMuCiAgICAgKiBAcmV0dXJuICAgICAgICAgICAgICBUaGUgdmFsdWUgcGFzc2VkIGluIGFzIHRvQXBwZW5kVG8gKHRoaXMgYWxsb3dzIGNoYWluaW5nLAogICAgICogICAgICAgICAgICAgICAgICAgICAgYXMgd2l0aCBVbmljb2RlU3RyaW5nOjphcHBlbmQoKSkKICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBVbmljb2RlU3RyaW5nJiBmb3JtYXQoY29uc3QgRm9ybWF0dGFibGUmIG9iaiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIHRvQXBwZW5kVG8sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBGaWVsZFBvc2l0aW9uJiBwb3MsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0OwoKICAgIC8qKgogICAgICogUmVkZWNsYXJlZCBGb3JtYXQgbWV0aG9kLgogICAgICogQHN0YWJsZQogICAgICovCiAgICBVbmljb2RlU3RyaW5nJiBmb3JtYXQoY29uc3QgRm9ybWF0dGFibGUmIG9iaiwKICAgICAgICAgICAgICAgICAgICAgICAgICBVbmljb2RlU3RyaW5nJiByZXN1bHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdDsKCiAgICAvKioKICAgICAqIFBhcnNlcyB0aGUgc3RyaW5nLgogICAgICogPFA+CiAgICAgKiBDYXZlYXRzOiBUaGUgcGFyc2UgbWF5IGZhaWwgaW4gYSBudW1iZXIgb2YgY2lyY3Vtc3RhbmNlcy4gIEZvcgogICAgICogZXhhbXBsZToKICAgICAqIDxQPgogICAgICogSWYgb25lIG9mIHRoZSBhcmd1bWVudHMgZG9lcyBub3Qgb2NjdXIgaW4gdGhlIHBhdHRlcm4uCiAgICAgKiA8UD4KICAgICAqIElmIHRoZSBmb3JtYXQgb2YgYW4gYXJndW1lbnQgaXMgbG9zZXMgaW5mb3JtYXRpb24sIHN1Y2ggYXMgd2l0aAogICAgICogYSBjaG9pY2UgZm9ybWF0IHdoZXJlIGEgbGFyZ2UgbnVtYmVyIGZvcm1hdHMgdG8gIm1hbnkiLgogICAgICogPFA+CiAgICAgKiBEb2VzIG5vdCB5ZXQgaGFuZGxlIHJlY3Vyc2lvbiAod2hlcmUgdGhlIHN1YnN0aXR1dGVkIHN0cmluZ3MKICAgICAqIGNvbnRhaW4ge259IHJlZmVyZW5jZXMuKQogICAgICogPFA+CiAgICAgKiBXaWxsIG5vdCBhbHdheXMgZmluZCBhIG1hdGNoIChvciB0aGUgY29ycmVjdCBtYXRjaCkgaWYgc29tZQogICAgICogcGFydCBvZiB0aGUgcGFyc2UgaXMgYW1iaWd1b3VzLiAgRm9yIGV4YW1wbGUsIGlmIHRoZSBwYXR0ZXJuCiAgICAgKiAiezF9LHsyfSIgaXMgdXNlZCB3aXRoIHRoZSBzdHJpbmcgYXJndW1lbnRzIHsiYSxiIiwgImMifSwgaXQKICAgICAqIHdpbGwgZm9ybWF0IGFzICJhLGIsYyIuICBXaGVuIHRoZSByZXN1bHQgaXMgcGFyc2VkLCBpdCB3aWxsCiAgICAgKiByZXR1cm4geyJhIiwgImIsYyJ9LgogICAgICogPFA+CiAgICAgKiBJZiBhIHNpbmdsZSBhcmd1bWVudCBpcyBmb3JtYXR0ZWQgdHdpY2UgaW4gdGhlIHN0cmluZywgdGhlbiB0aGUKICAgICAqIGxhdGVyIHBhcnNlIHdpbnMuCiAgICAgKgogICAgICogQHBhcmFtIHNvdXJjZSAgICBTdHJpbmcgdG8gYmUgcGFyc2VkLgogICAgICogQHBhcmFtIHN0YXR1cyAgICBPbiBpbnB1dCwgc3RhcnRpbmcgcG9zaXRpb24gZm9yIHBhcnNlLiBPbiBvdXRwdXQsCiAgICAgKiAgICAgICAgICAgICAgICAgIGZpbmFsIHBvc2l0aW9uIGFmdGVyIHBhcnNlLgogICAgICogQHBhcmFtIGNvdW50ICAgICBPdXRwdXQgcGFyYW0gdG8gcmVjZWl2ZSBzaXplIG9mIHJldHVybmVkIGFycmF5LgogICAgICogQHJlc3VsdCAgICAgICAgICBBcnJheSBvZiBGb3JtYXR0YWJsZSBvYmplY3RzLCB3aXRoIGxlbmd0aAogICAgICogICAgICAgICAgICAgICAgICAnY291bnQnLCBvd25lZCBieSB0aGUgY2FsbGVyLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIEZvcm1hdHRhYmxlKiBwYXJzZSggY29uc3QgVW5pY29kZVN0cmluZyYgc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFBhcnNlUG9zaXRpb24mIHN0YXR1cywKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90JiBjb3VudCkgY29uc3Q7CgogICAgLyoqCiAgICAgKiBQYXJzZXMgdGhlIHN0cmluZy4gRG9lcyBub3QgeWV0IGhhbmRsZSByZWN1cnNpb24gKHdoZXJlCiAgICAgKiB0aGUgc3Vic3RpdHV0ZWQgc3RyaW5ncyBjb250YWluIHtufSByZWZlcmVuY2VzLikKICAgICAqCiAgICAgKiBAcGFyYW0gc291cmNlICAgIFN0cmluZyB0byBiZSBwYXJzZWQuCiAgICAgKiBAcGFyYW0gY291bnQgICAgIE91dHB1dCBwYXJhbSB0byByZWNlaXZlIHNpemUgb2YgcmV0dXJuZWQgYXJyYXkuCiAgICAgKiBAcGFyYW0gc3RhdHVzICAgIE91dHB1dCBwYXJhbSB0byByZWNlaXZlIHN1Y2Nlc3MvZXJyb3IgY29kZS4KICAgICAqIEByZXN1bHQgICAgICAgICAgQXJyYXkgb2YgRm9ybWF0dGFibGUgb2JqZWN0cywgd2l0aCBsZW5ndGgKICAgICAqICAgICAgICAgICAgICAgICAgJ2NvdW50Jywgb3duZWQgYnkgdGhlIGNhbGxlci4KICAgICAqIEBzdGFibGUKICAgICAqLwogICAgdmlydHVhbCBGb3JtYXR0YWJsZSogcGFyc2UoIGNvbnN0IFVuaWNvZGVTdHJpbmcmIHNvdXJjZSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnQzMl90JiBjb3VudCwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdGF0dXMpIGNvbnN0OwoKICAgIC8qKgogICAgICogUGFyc2UgYSBzdHJpbmcgdG8gcHJvZHVjZSBhbiBvYmplY3QuICBUaGlzIG1ldGhvZHMgaGFuZGxlcwogICAgICogcGFyc2luZyBvZiBtZXNzYWdlIHN0cmluZ3MgaW50byBhcnJheXMgb2YgRm9ybWF0dGFibGUgb2JqZWN0cy4KICAgICAqIERvZXMgbm90IHlldCBoYW5kbGUgcmVjdXJzaW9uICh3aGVyZSB0aGUgc3Vic3RpdHV0ZWQgc3RyaW5ncwogICAgICogY29udGFpbiAlbiByZWZlcmVuY2VzLikKICAgICAqIDxQPgogICAgICogQmVmb3JlIGNhbGxpbmcsIHNldCBwYXJzZV9wb3MuaW5kZXggdG8gdGhlIG9mZnNldCB5b3Ugd2FudCB0bwogICAgICogc3RhcnQgcGFyc2luZyBhdCBpbiB0aGUgc291cmNlLiBBZnRlciBjYWxsaW5nLCBwYXJzZV9wb3MuaW5kZXgKICAgICAqIGlzIHRoZSBlbmQgb2YgdGhlIHRleHQgeW91IHBhcnNlZC4gIElmIGVycm9yIG9jY3VycywgaW5kZXggaXMKICAgICAqIHVuY2hhbmdlZC4KICAgICAqIDxQPgogICAgICogV2hlbiBwYXJzaW5nLCBsZWFkaW5nIHdoaXRlc3BhY2UgaXMgZGlzY2FyZGVkICh3aXRoIHN1Y2Nlc3NmdWwKICAgICAqIHBhcnNlKSwgd2hpbGUgdHJhaWxpbmcgd2hpdGVzcGFjZSBpcyBsZWZ0IGFzIGlzLgogICAgICogPFA+CiAgICAgKiBTZWUgRm9ybWF0OjpwYXJzZU9iamVjdCgpIGZvciBtb3JlLgogICAgICoKICAgICAqIEBwYXJhbSBzb3VyY2UgICAgVGhlIHN0cmluZyB0byBiZSBwYXJzZWQgaW50byBhbiBvYmplY3QuCiAgICAgKiBAcGFyYW0gcmVzdWx0ICAgIEZvcm1hdHRhYmxlIHRvIGJlIHNldCB0byB0aGUgcGFyc2UgcmVzdWx0LgogICAgICogICAgICAgICAgICAgICAgICBJZiBwYXJzZSBmYWlscywgcmV0dXJuIGNvbnRlbnRzIGFyZSB1bmRlZmluZWQuCiAgICAgKiBAcGFyYW0gcGFyc2VfcG9zIFRoZSBwb3NpdGlvbiB0byBzdGFydCBwYXJzaW5nIGF0LiBVcG9uIHJldHVybgogICAgICogICAgICAgICAgICAgICAgICB0aGlzIHBhcmFtIGlzIHNldCB0byB0aGUgcG9zaXRpb24gYWZ0ZXIgdGhlCiAgICAgKiAgICAgICAgICAgICAgICAgIGxhc3QgY2hhcmFjdGVyIHN1Y2Nlc3NmdWxseSBwYXJzZWQuIElmIHRoZQogICAgICogICAgICAgICAgICAgICAgICBzb3VyY2UgaXMgbm90IHBhcnNlZCBzdWNjZXNzZnVsbHksIHRoaXMgcGFyYW0KICAgICAqICAgICAgICAgICAgICAgICAgd2lsbCByZW1haW4gdW5jaGFuZ2VkLgogICAgICogQHJldHVybiAgICAgICAgICBBIG5ld2x5IGNyZWF0ZWQgRm9ybWF0dGFibGUqIG9iamVjdCwgb3IgTlVMTAogICAgICogICAgICAgICAgICAgICAgICBvbiBmYWlsdXJlLiAgVGhlIGNhbGxlciBvd25zIHRoaXMgYW5kIHNob3VsZAogICAgICogICAgICAgICAgICAgICAgICBkZWxldGUgaXQgd2hlbiBkb25lLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgcGFyc2VPYmplY3QoY29uc3QgVW5pY29kZVN0cmluZyYgc291cmNlLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgIEZvcm1hdHRhYmxlJiByZXN1bHQsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUGFyc2VQb3NpdGlvbiYgcGFyc2VfcG9zKSBjb25zdDsKCnB1YmxpYzoKICAgIC8qKgogICAgICogUmV0dXJucyBhIHVuaXF1ZSBjbGFzcyBJRCBQT0xZTU9SUEhJQ0FMTFkuICBQdXJlIHZpcnR1YWwgb3ZlcnJpZGUuCiAgICAgKiBUaGlzIG1ldGhvZCBpcyB0byBpbXBsZW1lbnQgYSBzaW1wbGUgdmVyc2lvbiBvZiBSVFRJLCBzaW5jZSBub3QgYWxsCiAgICAgKiBDKysgY29tcGlsZXJzIHN1cHBvcnQgZ2VudWluZSBSVFRJLiAgUG9seW1vcnBoaWMgb3BlcmF0b3I9PSgpIGFuZAogICAgICogY2xvbmUoKSBtZXRob2RzIGNhbGwgdGhpcyBtZXRob2QuCiAgICAgKgogICAgICogQHJldHVybiAgICAgICAgICBUaGUgY2xhc3MgSUQgZm9yIHRoaXMgb2JqZWN0LiBBbGwgb2JqZWN0cyBvZiBhCiAgICAgKiAgICAgICAgICAgICAgICAgIGdpdmVuIGNsYXNzIGhhdmUgdGhlIHNhbWUgY2xhc3MgSUQuICBPYmplY3RzIG9mCiAgICAgKiAgICAgICAgICAgICAgICAgIG90aGVyIGNsYXNzZXMgaGF2ZSBkaWZmZXJlbnQgY2xhc3MgSURzLgogICAgICogQHN0YWJsZQogICAgICovCiAgICB2aXJ0dWFsIFVDbGFzc0lEIGdldER5bmFtaWNDbGFzc0lEKHZvaWQpIGNvbnN0OwoKICAgIC8qKgogICAgICogUmV0dXJuIHRoZSBjbGFzcyBJRCBmb3IgdGhpcyBjbGFzcy4gIFRoaXMgaXMgdXNlZnVsIG9ubHkgZm9yCiAgICAgKiBjb21wYXJpbmcgdG8gYSByZXR1cm4gdmFsdWUgZnJvbSBnZXREeW5hbWljQ2xhc3NJRCgpLiAgRm9yIGV4YW1wbGU6CiAgICAgKiA8cHJlPgogICAgICogLiAgIEJhc2UqIHBvbHltb3JwaGljX3BvaW50ZXIgPSBjcmVhdGVQb2x5bW9ycGhpY09iamVjdCgpOwogICAgICogLiAgIGlmIChwb2x5bW9ycGhpY19wb2ludGVyLT5nZXREeW5hbWljQ2xhc3NJRCgpID09CiAgICAgKiAuICAgICAgRGVyaXZlZDo6Z2V0U3RhdGljQ2xhc3NJRCgpKSAuLi4KICAgICAqIDwvcHJlPgogICAgICogQHJldHVybiAgICAgICAgICBUaGUgY2xhc3MgSUQgZm9yIGFsbCBvYmplY3RzIG9mIHRoaXMgY2xhc3MuCiAgICAgKiBAc3RhYmxlCiAgICAgKi8KICAgIHN0YXRpYyBVQ2xhc3NJRCBnZXRTdGF0aWNDbGFzc0lEKHZvaWQpIHsgcmV0dXJuIChVQ2xhc3NJRCkmZmdDbGFzc0lEOyB9Cgpwcml2YXRlOgogICAgc3RhdGljIGNoYXIgZmdDbGFzc0lEOwogICAgc3RhdGljIE51bWJlckZvcm1hdCogZmdOdW1iZXJGb3JtYXQ7CgogICAgLy8gZmdOdW1iZXJGb3JtYXQgaXMgaGVsZCBpbiBhIGNhY2hlIG9mIG9uZS4KCiAgICBzdGF0aWMgTnVtYmVyRm9ybWF0KiBnZXROdW1iZXJGb3JtYXQoVUVycm9yQ29kZSAmc3RhdHVzKTsgLy8gY2FsbCB0aGlzIGZ1bmN0aW9uIHRvICdjaGVjayBvdXQnIGEgbnVtYmVyZm9ybWF0IGZyb20gdGhlIGNhY2hlLgogICAgc3RhdGljIHZvaWQgICAgICAgICAgcmVsZWFzZU51bWJlckZvcm1hdChOdW1iZXJGb3JtYXQgKmFkb3B0KTsgLy8gY2FsbCB0aGlzIGZ1bmN0aW9uIHRvICdyZXR1cm4nIHRoZSBudW1iZXIgZm9ybWF0IHRvIHRoZSBjYWNoZS4KCiAgICBMb2NhbGUgICAgICAgICAgICAgICAgIGZMb2NhbGU7CiAgICBVbmljb2RlU3RyaW5nICAgICAgICAgZlBhdHRlcm47CiAgICAvLyBsYXRlciwgYWxsb3cgbW9yZSB0aGFuIHRlbiBpdGVtcwogICAgRm9ybWF0ICAgICAgICAgICAgICAgICAqZkZvcm1hdHNba01heEZvcm1hdF07CiAgICBpbnQzMl90ICAgICAgICAgICAgICpmT2Zmc2V0czsKICAgIGludDMyX3QgICAgICAgICAgICAgZkNvdW50OwogICAgaW50MzJfdCAgICAgICAgICAgICAqZkFyZ3VtZW50TnVtYmVyczsKICAgIGludDMyX3QgICAgICAgICAgICAgZk1heE9mZnNldDsKICAgIAogICAgLyoqCiAgICAgKiBJbnRlcm5hbCByb3V0aW5lIHVzZWQgYnkgZm9ybWF0LgogICAgICogQHBhcmFtIHJlY3Vyc2lvblByb3RlY3Rpb24gSW5pdGlhbGx5IHplcm8uIEJpdHMgMC4uOSBhcmUgdXNlZCB0byBpbmRpY2F0ZQogICAgICogdGhhdCBhIHBhcmFtZXRlciBoYXMgYWxyZWFkeSBiZWVuIHNlZW4sIHRvIGF2b2lkIHJlY3Vyc2lvbi4gIEN1cnJlbnRseQogICAgICogdW51c2VkLgogICAgICovCiAgICBzdGF0aWMgY29uc3QgaW50MzJfdCAgICAgICAgIGZnTGlzdExlbmd0aDsKICAgIHN0YXRpYyBjb25zdCBVbmljb2RlU3RyaW5nICAgICBmZ1R5cGVMaXN0W107CiAgICBzdGF0aWMgY29uc3QgVW5pY29kZVN0cmluZyAgICAgZmdNb2RpZmllckxpc3RbXTsKICAgIHN0YXRpYyBjb25zdCBVbmljb2RlU3RyaW5nICAgICBmZ0RhdGVNb2RpZmllckxpc3RbXTsKCiAgICAvKiogCiAgICAgKiBGaW5kcyB0aGUgd29yZCBzLCBpbiB0aGUga2V5d29yZCBsaXN0IGFuZCByZXR1cm5zIHRoZSBsb2NhdGVkIGluZGV4LgogICAgICogQHBhcmFtIHMgdGhlIGtleXdvcmQgdG8gYmUgc2VhcmNoZWQgZm9yLgogICAgICogQHBhcmFtIGxpc3QgdGhlIGxpc3Qgb2Yga2V5d29yZHMgdG8gYmUgc2VhcmNoZWQgd2l0aC4KICAgICAqIEByZXR1cm4gdGhlIGluZGV4IG9mIHRoZSBsaXN0IHdoaWNoIG1hdGNoZXMgdGhlIGtleXdvcmQgcy4KICAgICAqLwogICAgc3RhdGljIGludDMyX3QgZmluZEtleXdvcmQoIGNvbnN0IFVuaWNvZGVTdHJpbmcmIHMsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgVW5pY29kZVN0cmluZyogbGlzdCk7CgogICAgLyoqCiAgICAgKiBGb3JtYXRzIHRoZSBhcnJheSBvZiBhcmd1bWVudHMgYW5kIGNvcGllcyB0aGUgcmVzdWx0IGludG8gdGhlIHJlc3VsdCBidWZmZXIsCiAgICAgKiB1cGRhdGVzIHRoZSBmaWVsZCBwb3NpdGlvbi4KICAgICAqIEBwYXJhbSBhcmd1bWVudHMgdGhlIGZvcm1hdHRhYmxlIG9iamVjdHMgYXJyYXkuCiAgICAgKiBAcGFyYW0gY250IHRoZSBhcnJheSBjb3VudC4KICAgICAqIEBwYXJhbSBzdGF0dXMgZmllbGQgcG9zaXRpb24gc3RhdHVzLgogICAgICogQHBhcmFtIHJlY3Vyc2lvblByb3RlY3Rpb24gSW5pdGlhbGx5IHplcm8uIEJpdHMgMC4uOSBhcmUgdXNlZCB0byBpbmRpY2F0ZQogICAgICogdGhhdCBhIHBhcmFtZXRlciBoYXMgYWxyZWFkeSBiZWVuIHNlZW4sIHRvIGF2b2lkIHJlY3Vyc2lvbi4gIEN1cnJlbnRseQogICAgICogdW51c2VkLgogICAgICogQHBhcmFtIHN1Y2Nlc3MgdGhlIGVycm9yIGNvZGUgc3RhdHVzLgogICAgICovCiAgICBVbmljb2RlU3RyaW5nJiAgZm9ybWF0KCBjb25zdCBGb3JtYXR0YWJsZSogYXJndW1lbnRzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGludDMyX3QgY250LCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcmIHJlc3VsdCwgCiAgICAgICAgICAgICAgICAgICAgICAgICAgICBGaWVsZFBvc2l0aW9uJiBzdGF0dXMsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCByZWN1cnNpb25Qcm90ZWN0aW9uLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3VjY2VzcykgY29uc3Q7CgogICAgLyoqCiAgICAgKiBDaGVja3MgdGhlIHNlZ21lbnRzIGZvciB0aGUgY2xvc2VzdCBtYXRjaGVkIGZvcm1hdCBpbnN0YW5jZSBhbmQKICAgICAqIHVwZGF0ZXMgdGhlIGZvcm1hdCBhcnJheSB3aXRoIHRoZSBuZXcgZm9ybWF0IGluc3RhbmNlLgogICAgICogQHBhcmFtIHBvc2l0aW9uIHRoZSBsYXN0IHByb2Nlc3NlZCBvZmZzZXQgaW4gdGhlIHBhdHRlcm4gCiAgICAgKiBAcGFyYW0gb2Zmc2V0TnVtYmVyIHRoZSBvZmZzZXQgbnVtYmVyIG9mIHRoZSBsYXN0IHByb2Nlc3NlZCBzZWdtZW50CiAgICAgKiBAcGFyYW0gc2VnbWVudHMgdGhlIHN0cmluZyB0aGF0IGNvbnRhaW5zIHRoZSBwYXJzZWQgcGF0dGVybiBzZWdtZW50cy4KICAgICAqIEBwYXJhbSBzdWNjZXNzIHRoZSBlcnJvciBjb2RlCiAgICAgKi8KICAgIHZvaWQgICAgICAgICAgICBtYWtlRm9ybWF0KCAvKmludDMyX3QgcG9zaXRpb24sICovCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaW50MzJfdCBvZmZzZXROdW1iZXIsIAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFVuaWNvZGVTdHJpbmcqIHNlZ21lbnRzLCAKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlJiBzdWNjZXNzKTsKCiAgICAvKioKICAgICAqIENvbnZlbmllbmNlIG1ldGhvZCB0aGF0IG91Z2h0IHRvIGJlIGluIE51bWJlckZvcm1hdAogICAgICovCiAgICBOdW1iZXJGb3JtYXQqIGNyZWF0ZUludGVnZXJGb3JtYXQoY29uc3QgTG9jYWxlJiBsb2NhbGUsIFVFcnJvckNvZGUmIHN0YXR1cykgY29uc3Q7CgogICAgLyoqCiAgICAgKiBDaGVja3MgdGhlIHJhbmdlIG9mIHRoZSBzb3VyY2UgdGV4dCB0byBxdW90ZSB0aGUgc3BlY2lhbAogICAgICogY2hhcmFjdGVycywgeyBhbmQgJyBhbmQgY29weSB0byB0YXJnZXQgYnVmZmVyLgogICAgICogQHBhcmFtIHNvdXJjZQogICAgICogQHBhcmFtIHN0YXJ0IHRoZSB0ZXh0IG9mZnNldCB0byBzdGFydCB0aGUgcHJvY2VzcyBvZiBpbiB0aGUgc291cmNlIHN0cmluZwogICAgICogQHBhcmFtIGVuZCB0aGUgdGV4dCBvZmZzZXQgdG8gZW5kIHRoZSBwcm9jZXNzIG9mIGluIHRoZSBzb3VyY2Ugc3RyaW5nCiAgICAgKiBAcGFyYW0gdGFyZ2V0IHRoZSByZXN1bHQgYnVmZmVyCiAgICAgKi8KICAgIHN0YXRpYyB2b2lkIGNvcHlBbmRGaXhRdW90ZXMoY29uc3QgVW5pY29kZVN0cmluZyYgc291cmNlLCBpbnQzMl90IHN0YXJ0LCBpbnQzMl90IGVuZCwgVW5pY29kZVN0cmluZyYgdGFyZ2V0KTsKCiAgICAvKioKICAgICAqIENvbnZlcnRzIGEgc3RyaW5nIHRvIGFuIGludGVnZXIgdmFsdWUgdXNpbmcgYSBkZWZhdWx0IE51bWJlckZvcm1hdCBvYmplY3QKICAgICAqIHdoaWNoIGlzIHN0YXRpYyAoc2hhcmVkIGJ5IGFsbCBNZXNzYWdlRm9ybWF0IGluc3RhbmNlcykuICBUaGlzIHJlcGxhY2VzCiAgICAgKiBhIGNhbGwgdG8gd3RvaSgpLgogICAgICogQHBhcmFtIHN0cmluZyB0aGUgc291cmNlIHN0cmluZyB0byBjb252ZXJ0IHdpdGgKICAgICAqIEBwYXJhbSBzdGF0dXMgdGhlIGVycm9yIGNvZGUuCiAgICAgKiBAcmV0dXJuIHRoZSBjb252ZXJ0ZWQgbnVtYmVyLgogICAgICovCiAgICBzdGF0aWMgaW50MzJfdCBzdG9pKGNvbnN0IFVuaWNvZGVTdHJpbmcmIHN0cmluZywgVUVycm9yQ29kZSYgc3RhdHVzKTsKCiAgICAvKioKICAgICAqIENvbnZlcnRzIGFuIGludGVnZXIgdmFsdWUgdG8gYSBzdHJpbmcgdXNpbmcgYSBkZWZhdWx0IE51bWJlckZvcm1hdCBvYmplY3QKICAgICAqIHdoaWNoIGlzIHN0YXRpYyAoc2hhcmVkIGJ5IGFsbCBNZXNzYWdlRm9ybWF0IGluc3RhbmNlcykuICBUaGlzIHJlcGxhY2VzCiAgICAgKiBhIGNhbGwgdG8gd3RvaSgpLgogICAgICogQHBhcmFtIGkgdGhlIGludGVnZXIgdG8gZm9ybWF0CiAgICAgKiBAcGFyYW0gc3RyaW5nIHRoZSBkZXN0aW5hdGlvbiBzdHJpbmcKICAgICAqIEByZXR1cm4gYSByZWZlcmVuY2UgdG8gc3RyaW5nLgogICAgICovCiAgICBzdGF0aWMgVW5pY29kZVN0cmluZyYgaXRvcyhpbnQzMl90IGksIFVuaWNvZGVTdHJpbmcmIHN0cmluZyk7Cn07CiAKaW5saW5lIFVDbGFzc0lEIApNZXNzYWdlRm9ybWF0OjpnZXREeW5hbWljQ2xhc3NJRCgpIGNvbnN0CnsgCiAgICByZXR1cm4gTWVzc2FnZUZvcm1hdDo6Z2V0U3RhdGljQ2xhc3NJRCgpOyAKfQoKaW5saW5lIFVuaWNvZGVTdHJpbmcmCk1lc3NhZ2VGb3JtYXQ6OmZvcm1hdChjb25zdCBGb3JtYXR0YWJsZSYgb2JqLAogICAgICAgICAgICAgICAgICAgICAgVW5pY29kZVN0cmluZyYgcmVzdWx0LAogICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSYgc3RhdHVzKSBjb25zdCB7CiAgICByZXR1cm4gRm9ybWF0Ojpmb3JtYXQob2JqLCByZXN1bHQsIHN0YXR1cyk7Cn0KCiNlbmRpZiAvLyBfTVNHRk1UCi8vZW9mCg==