LyoKKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKgoqICAgQ29weXJpZ2h0IChDKSAyMDAxIElCTSBhbmQgb3RoZXJzLiBBbGwgcmlnaHRzIHJlc2VydmVkLgoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqCiogICBEYXRlICAgICAgICBOYW1lICAgICAgICBEZXNjcmlwdGlvbgoqICAwMy8yMi8yMDAwICAgaGVsZW5hICAgICAgQ3JlYXRpb24uCioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioKKi8KCiNpZm5kZWYgU1RTRUFSQ0hfSAojZGVmaW5lIFNUU0VBUkNIX0gKCiNpbmNsdWRlICJ1bmljb2RlL3RibGNvbGwuaCIKI2luY2x1ZGUgInVuaWNvZGUvY29sZWl0ci5oIgojaW5jbHVkZSAidW5pY29kZS9zZWFyY2guaCIKClVfTkFNRVNQQUNFX0JFR0lOCgovKioKICogPHR0PlN0cmluZ1NlYXJjaDwvdHQ+IGlzIGEgPHR0PlNlYXJjaEl0ZXJhdG9yPC90dD4gdGhhdCBwcm92aWRlcwogKiBsYW5ndWFnZS1zZW5zaXRpdmUgdGV4dCBzZWFyY2hpbmcgYmFzZWQgb24gdGhlIGNvbXBhcmlzb24gcnVsZXMgZGVmaW5lZAogKiBpbiBhIHtAbGluayBSdWxlQmFzZWRDb2xsYXRvcn0gb2JqZWN0LgogKiBTdHJpbmdTZWFyY2ggZW5zdXJlcyB0aGF0IGxhbmd1YWdlIGVjY2VudHJpY2l0eSBjYW4gYmUgCiAqIGhhbmRsZWQsIGUuZy4gZm9yIHRoZSBHZXJtYW4gY29sbGF0b3IsIGNoYXJhY3RlcnMg3yBhbmQgU1Mgd2lsbCBiZSBtYXRjaGVkIAogKiBpZiBjYXNlIGlzIGNob3NlbiB0byBiZSBpZ25vcmVkLgogKiBTZWUgdGhlIDxhIGhyZWY9aHR0cDovL29zcy5zb2Z0d2FyZS5pYm0uY29tL2ljdS9kZXZlbG9wL2NvbGxhdGlvbi9JQ1VfY29sbGF0aW9uX2Rlc2lnbi5odG0+CiAqICJJQ1UgQ29sbGF0aW9uIERlc2lnbiBEb2N1bWVudCI8L2E+IGZvciBtb3JlIGluZm9ybWF0aW9uLgogKiA8cD4gCiAqIFRoZSBhbGdvcml0aG0gaW1wbGVtZW50ZWQgaXMgYSBtb2RpZmllZCBmb3JtIG9mIHRoZSBCb3llciBNb29yZSdzIHNlYXJjaC4KICogRm9yIG1vcmUgaW5mb3JtYXRpb24gIHNlZSAKICogPGEgaHJlZj1odHRwOi8vb3NzLnNvZnR3YXJlLmlibS5jb20vaWN1L2RvY3MvcGFwZXJzL3RleHQtc2VhcmNoLmh0bWw+CiAqICJFZmZpY2llbnQgVGV4dCBTZWFyY2hpbmcgaW4gSmF2YSI8L2E+LCBwdWJsaXNoZWQgaW4gPGk+SmF2YSBSZXBvcnQ8L2k+IAogKiBpbiBGZWJydWFyeSwgMTk5OSwgZm9yIGZ1cnRoZXIgaW5mb3JtYXRpb24gb24gdGhlIGFsZ29yaXRobS4KICogPHA+CiAqIFRoZXJlIGFyZSAyIG1hdGNoIG9wdGlvbnMgZm9yIHNlbGVjdGlvbjo8YnI+CiAqIExldCBTJyBiZSB0aGUgc3ViLXN0cmluZyBvZiBhIHRleHQgc3RyaW5nIFMgYmV0d2VlbiB0aGUgb2Zmc2V0cyBzdGFydCBhbmQgCiAqIGVuZCA8c3RhcnQsIGVuZD4uCiAqIDxicj4KICogQSBwYXR0ZXJuIHN0cmluZyBQIG1hdGNoZXMgYSB0ZXh0IHN0cmluZyBTIGF0IHRoZSBvZmZzZXRzIDxzdGFydCwgZW5kPiAKICogaWYKICogPHByZT4gCiAqIG9wdGlvbiAxLiBTb21lIGNhbm9uaWNhbCBlcXVpdmFsZW50IG9mIFAgbWF0Y2hlcyBzb21lIGNhbm9uaWNhbCBlcXVpdmFsZW50IAogKiAgICAgICAgICAgb2YgUycKICogb3B0aW9uIDIuIFAgbWF0Y2hlcyBTJyBhbmQgaWYgUCBzdGFydHMgb3IgZW5kcyB3aXRoIGEgY29tYmluaW5nIG1hcmssIAogKiAgICAgICAgICAgdGhlcmUgZXhpc3RzIG5vIG5vbi1pZ25vcmFibGUgY29tYmluaW5nIG1hcmsgYmVmb3JlIG9yIGFmdGVyIFM/IAogKiAgICAgICAgICAgaW4gUyByZXNwZWN0aXZlbHkuIAogKiA8L3ByZT4KICogT3B0aW9uIDIuIHdpbGwgYmUgdGhlIGRlZmF1bHS3CiAqIDxwPgogKiBUaGlzIHNlYXJjaCBoYXMgQVBJcyBzaW1pbGFyIHRvIHRoYXQgb2Ygb3RoZXIgdGV4dCBpdGVyYXRpb24gbWVjaGFuaXNtcyAKICogc3VjaCBhcyB0aGUgYnJlYWsgaXRlcmF0b3JzIGluIDx0dD5CcmVha0l0ZXJhdG9yPC90dD4uIFVzaW5nIHRoZXNlIAogKiBBUElzLCBpdCBpcyBlYXN5IHRvIHNjYW4gdGhyb3VnaCB0ZXh0IGxvb2tpbmcgZm9yIGFsbCBvY2N1cmFuY2VzIG9mIAogKiBhIGdpdmVuIHBhdHRlcm4uIFRoaXMgc2VhcmNoIGl0ZXJhdG9yIGFsbG93cyBjaGFuZ2luZyBvZiBkaXJlY3Rpb24gYnkgCiAqIGNhbGxpbmcgYSA8dHQ+cmVzZXQ8L3R0PiBmb2xsb3dlZCBieSBhIDx0dD5uZXh0PC90dD4gb3IgPHR0PnByZXZpb3VzPC90dD4uIAogKiBUaG91Z2ggYSBkaXJlY3Rpb24gY2hhbmdlIGNhbiBvY2N1ciB3aXRob3V0IGNhbGxpbmcgPHR0PnJlc2V0PC90dD4gZmlyc3QsICAKICogdGhpcyBvcGVyYXRpb24gY29tZXMgd2l0aCBzb21lIHNwZWVkIHBlbmFsdHkuCiAqIE1hdGNoIHJlc3VsdHMgaW4gdGhlIGZvcndhcmQgZGlyZWN0aW9uIHdpbGwgbWF0Y2ggdGhlIHJlc3VsdCBtYXRjaGVzIGluIAogKiB0aGUgYmFja3dhcmRzIGRpcmVjdGlvbiBpbiB0aGUgcmV2ZXJzZSBvcmRlcgogKiA8cD4KICogPHR0PlNlYXJjaEl0ZXJhdG9yPC90dD4gcHJvdmlkZXMgQVBJcyB0byBzcGVjaWZ5IHRoZSBzdGFydGluZyBwb3NpdGlvbiAKICogd2l0aGluIHRoZSB0ZXh0IHN0cmluZyB0byBiZSBzZWFyY2hlZCwgZS5nLiA8dHQ+c2V0T2Zmc2V0PC90dD4sCiAqIDx0dD5wcmVjZWRpbmc8L3R0PiBhbmQgPHR0PmZvbGxvd2luZzwvdHQ+LiBTaW5jZSB0aGUgCiAqIHN0YXJ0aW5nIHBvc2l0aW9uIHdpbGwgYmUgc2V0IGFzIGl0IGlzIHNwZWNpZmllZCwgcGxlYXNlIHRha2Ugbm90ZSB0aGF0IAogKiB0aGVyZSBhcmUgc29tZSBkYW5nZXIgcG9pbnRzIHdoaWNoIHRoZSBzZWFyY2ggbWF5IHJlbmRlciBpbmNvcnJlY3QgCiAqIHJlc3VsdHM6CiAqIDx1bD4KICogPGxpPiBUaGUgbWlkc3Qgb2YgYSBzdWJzdHJpbmcgdGhhdCByZXF1aXJlcyBub3JtYWxpemF0aW9uLgogKiA8bGk+IElmIHRoZSBmb2xsb3dpbmcgbWF0Y2ggaXMgdG8gYmUgZm91bmQsIHRoZSBwb3NpdGlvbiBzaG91bGQgbm90IGJlIHRoZQogKiAgICAgIHNlY29uZCBjaGFyYWN0ZXIgd2hpY2ggcmVxdWlyZXMgdG8gYmUgc3dhcHBlZCB3aXRoIHRoZSBwcmVjZWRpbmcgCiAqICAgICAgY2hhcmFjdGVyLiBWaWNlIHZlcnNhLCBpZiB0aGUgcHJlY2VkaW5nIG1hdGNoIGlzIHRvIGJlIGZvdW5kLCAKICogICAgICBwb3NpdGlvbiB0byBzZWFyY2ggZnJvbSBzaG91bGQgbm90IGJlIHRoZSBmaXJzdCBjaGFyYWN0ZXIgd2hpY2ggCiAqICAgICAgcmVxdWlyZXMgdG8gYmUgc3dhcHBlZCB3aXRoIHRoZSBuZXh0IGNoYXJhY3Rlci4gRS5nIGNlcnRhaW4gVGhhaSBhbmQKICogICAgICBMYW8gY2hhcmFjdGVycyByZXF1aXJlIHN3YXBwaW5nLgogKiA8bGk+IElmIGEgZm9sbG93aW5nIHBhdHRlcm4gbWF0Y2ggaXMgdG8gYmUgZm91bmQsIGFueSBwb3NpdGlvbiB3aXRoaW4gYSAKICogICAgICBjb250cmFjdGluZyBzZXF1ZW5jZSBleGNlcHQgdGhlIGZpcnN0IHdpbGwgZmFpbC4gVmljZSB2ZXJzYSBpZiBhIAogKiAgICAgIHByZWNlZGluZyBwYXR0ZXJuIG1hdGNoIGlzIHRvIGJlIGZvdW5kLCBhIGludmFsaWQgc3RhcnRpbmcgcG9pbnQgCiAqICAgICAgd291bGQgYmUgYW55IGNoYXJhY3RlciB3aXRoaW4gYSBjb250cmFjdGluZyBzZXF1ZW5jZSBleGNlcHQgdGhlIGxhc3QuCiAqIDxcdWw+CiAqIDxwPgogKiBBIGJyZWFraXRlcmF0b3IgY2FuIGJlIHVzZWQgaWYgb25seSBtYXRjaGVzIGF0IGxvZ2ljYWwgYnJlYWtzIGFyZSBkZXNpcmVkLgogKiBVc2luZyBhIGJyZWFraXRlcmF0b3Igd2lsbCBvbmx5IGdpdmUgeW91IHJlc3VsdHMgdGhhdCBleGFjdGx5IG1hdGNoZXMgdGhlCiAqIGJvdW5kYXJpZXMgZ2l2ZW4gYnkgdGhlIGJyZWFraXRlcmF0b3IuIEZvciBpbnN0YW5jZSB0aGUgcGF0dGVybiAiZSIgd2lsbAogKiBub3QgYmUgZm91bmQgaW4gdGhlIHN0cmluZyAiXHUwMGU5IiBpZiBhIGNoYXJhY3RlciBicmVhayBpdGVyYXRvciBpcyB1c2VkLgogKiA8cD4KICogT3B0aW9ucyBhcmUgcHJvdmlkZWQgdG8gaGFuZGxlIG92ZXJsYXBwaW5nIG1hdGNoZXMuIAogKiBFLmcuIEluIEVuZ2xpc2gsIG92ZXJsYXBwaW5nIG1hdGNoZXMgcHJvZHVjZXMgdGhlIHJlc3VsdCAwIGFuZCAyIAogKiBmb3IgdGhlIHBhdHRlcm4gImFiYWIiIGluIHRoZSB0ZXh0ICJhYmFiYWIiLCB3aGVyZSBlbHNlIG11dHVhbGx5IAogKiBleGNsdXNpdmUgbWF0Y2hlcyBvbmx5IHByb2R1Y2UgdGhlIHJlc3VsdCBvZiAwLgogKiA8cD4KICogVGhvdWdoIGNvbGxhdG9yIGF0dHJpYnV0ZXMgd2lsbCBiZSB0YWtlbiBpbnRvIGNvbnNpZGVyYXRpb24gd2hpbGUgCiAqIHBlcmZvcm1pbmcgbWF0Y2hlcywgdGhlcmUgYXJlIG5vIEFQSXMgaGVyZSBmb3Igc2V0dGluZyBhbmQgZ2V0dGluZyB0aGUgCiAqIGF0dHJpYnV0ZXMuIFRoZXNlIGF0dHJpYnV0ZXMgY2FuIGJlIHNldCBieSBnZXR0aW5nIHRoZSBjb2xsYXRvcgogKiBmcm9tIDx0dD5nZXRDb2xsYXRvcjwvdHQ+IGFuZCB1c2luZyB0aGUgQVBJcyBpbiA8dHQ+Y29sbC5oPC90dD4uCiAqIExhc3RseSB0byB1cGRhdGUgU3RyaW5nU2VhcmNoIHRvIHRoZSBuZXcgY29sbGF0b3IgYXR0cmlidXRlcywgCiAqIHJlc2V0KCkgaGFzIHRvIGJlIGNhbGxlZC4KICogPHA+IAogKiBSZXN0cmljdGlvbjogPGJyPgogKiBDdXJyZW50bHkgdGhlcmUgYXJlIG5vIGNvbXBvc2l0ZSBjaGFyYWN0ZXJzIHRoYXQgY29uc2lzdHMgb2YgYQogKiBjaGFyYWN0ZXIgd2l0aCBjb21iaW5pbmcgY2xhc3MgPiAwIGJlZm9yZSBhIGNoYXJhY3RlciB3aXRoIGNvbWJpbmluZyAKICogY2xhc3MgPT0gMC4gSG93ZXZlciwgaWYgc3VjaCBhIGNoYXJhY3RlciBleGlzdHMgaW4gdGhlIGZ1dHVyZSwgIAogKiBTdHJpbmdTZWFyY2ggZG9lcyBub3QgZ3VhcmFudGVlIHRoZSByZXN1bHRzIGZvciBvcHRpb24gMS4KICogPHA+CiAqIENvbnN1bHQgdGhlIDx0dD5TZWFyY2hJdGVyYXRvcjwvdHQ+IGRvY3VtZW50YXRpb24gZm9yIGluZm9ybWF0aW9uIG9uCiAqIGFuZCBleGFtcGxlcyBvZiBob3cgdG8gdXNlIGluc3RhbmNlcyBvZiB0aGlzIGNsYXNzIHRvIGltcGxlbWVudCB0ZXh0CiAqIHNlYXJjaGluZy4KICogPHByZT48Y29kZT4KICogVW5pY29kZVN0cmluZyB0YXJnZXQoIlRoZSBxdWljayBicm93biBmb3gganVtcGVkIG92ZXIgdGhlIGxhenkgZm94Iik7CiAqIFVuaWNvZGVTdHJpbmcgcGF0dGVybigiZm94Iik7CiAqCiAqIFNlYXJjaEl0ZXJhdG9yICppdGVyICA9IG5ldyBTdHJpbmdTZWFyY2gocGF0dGVybiwgdGFyZ2V0KTsKICogVUVycm9yQ29kZSAgICAgIGVycm9yID0gVV9aRVJPX0VSUk9SOwogKiBmb3IgKGludCBwb3MgPSBpdGVyLT5maXJzdChlcnJvcik7IHBvcyAhPSBVU0VBUkNIX0RPTkU7IAogKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBwb3MgPSBpdGVyLT5uZXh0KGVycm9yKSkgewogKiAgICAgcHJpbnRmKCJGb3VuZCBtYXRjaCBhdCAlZCBwb3MsIGxlbmd0aCBpcyAlZFxuIiwgcG9zLCAKICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpdGVyLmdldE1hdGNoTGVuZ3RoKCkpOwogKiB9CiAqIDwvY29kZT48L3ByZT4KICogQHNlZSBTZWFyY2hJdGVyYXRvcgogKiBAc2VlIFJ1bGVCYXNlZENvbGxhdG9yCiAqIEBzaW5jZSBJQ1UgMi4wCiAqLwoKY2xhc3MgVV9JMThOX0FQSSBTdHJpbmdTZWFyY2ggOiBwdWJsaWMgU2VhcmNoSXRlcmF0b3IKewpwdWJsaWM6CgogICAgLy8gcHVibGljIGNvbnN0cnVjdG9ycyBhbmQgZGVzdHJ1Y3RvcnMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiAgICAvKioKICAgICAqIENyZWF0aW5nIGEgPHR0PlN0cmluZ1NlYXJjaDwvdHQ+IGluc3RhbmNlIHVzaW5nIHRoZSBhcmd1bWVudCBsb2NhbGUgCiAgICAgKiBsYW5ndWFnZSBydWxlIHNldC4gQSBjb2xsYXRvciB3aWxsIGJlIGNyZWF0ZWQgaW4gdGhlIHByb2Nlc3MsIHdoaWNoIAogICAgICogd2lsbCBiZSBvd25lZCBieSB0aGlzIGluc3RhbmNlIGFuZCB3aWxsIGJlIGRlbGV0ZWQgaW4gZHVyaW5nIAogICAgICogZGVzdHJ1Y3Rpb24KICAgICAqIEBwYXJhbSBwYXR0ZXJuIFRoZSB0ZXh0IGZvciB3aGljaCB0aGlzIG9iamVjdCB3aWxsIHNlYXJjaC4KICAgICAqIEBwYXJhbSB0ZXh0ICAgIFRoZSB0ZXh0IGluIHdoaWNoIHRvIHNlYXJjaCBmb3IgdGhlIHBhdHRlcm4uCiAgICAgKiBAcGFyYW0gbG9jYWxlICBBIGxvY2FsZSB3aGljaCBkZWZpbmVzIHRoZSBsYW5ndWFnZS1zZW5zaXRpdmUgCiAgICAgKiAgICAgICAgICAgICAgICBjb21wYXJpc29uIHJ1bGVzIHVzZWQgdG8gZGV0ZXJtaW5lIHdoZXRoZXIgdGV4dCBpbiB0aGUgCiAgICAgKiAgICAgICAgICAgICAgICBwYXR0ZXJuIGFuZCB0YXJnZXQgbWF0Y2hlcy4gCiAgICAgKiBAcGFyYW0gYnJlYWtpdGVyIEEgPHR0PkJyZWFrSXRlcmF0b3I8L3R0PiBvYmplY3QgdXNlZCB0byBjb25zdHJhaW4gCiAgICAgKiAgICAgICAgICAgICAgICB0aGUgbWF0Y2hlcyB0aGF0IGFyZSBmb3VuZC4gTWF0Y2hlcyB3aG9zZSBzdGFydCBhbmQgZW5kIAogICAgICogICAgICAgICAgICAgICAgaW5kaWNlcyBpbiB0aGUgdGFyZ2V0IHRleHQgYXJlIG5vdCBib3VuZGFyaWVzIGFzIAogICAgICogICAgICAgICAgICAgICAgZGV0ZXJtaW5lZCBieSB0aGUgPHR0PkJyZWFrSXRlcmF0b3I8L3R0PiBhcmUgCiAgICAgKiAgICAgICAgICAgICAgICBpZ25vcmVkLiBJZiB0aGlzIGJlaGF2aW9yIGlzIG5vdCBkZXNpcmVkLCAKICAgICAqICAgICAgICAgICAgICAgIDx0dD5OVUxMPC90dD4gY2FuIGJlIHBhc3NlZCBpbiBpbnN0ZWFkLgogICAgICogQHBhcmFtIHN0YXR1cyAgZm9yIGVycm9ycyBpZiBhbnkuIElmIHBhdHRlcm4gb3IgdGV4dCBpcyBOVUxMLCBvciBpZgogICAgICogICAgICAgICAgICAgICBlaXRoZXIgdGhlIGxlbmd0aCBvZiBwYXR0ZXJuIG9yIHRleHQgaXMgMCB0aGVuIGFuIAogICAgICogICAgICAgICAgICAgICBVX0lMTEVHQUxfQVJHVU1FTlRfRVJST1IgaXMgcmV0dXJuZWQuCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICBTdHJpbmdTZWFyY2goY29uc3QgVW5pY29kZVN0cmluZyAmcGF0dGVybiwgY29uc3QgVW5pY29kZVN0cmluZyAmdGV4dCwKICAgICAgICAgICAgICAgICBjb25zdCBMb2NhbGUgICAgICAgICZsb2NhbGUsICAgICAgIAogICAgICAgICAgICAgICAgICAgICAgIEJyZWFrSXRlcmF0b3IgKmJyZWFraXRlciwKICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICAgICZzdGF0dXMpOwoKICAgIC8qKgogICAgICogQ3JlYXRpbmcgYSA8dHQ+U3RyaW5nU2VhcmNoPC90dD4gaW5zdGFuY2UgdXNpbmcgdGhlIGFyZ3VtZW50IGNvbGxhdG9yIAogICAgICogbGFuZ3VhZ2UgcnVsZSBzZXQuIE5vdGUsIHVzZXIgcmV0YWlucyB0aGUgb3duZXJzaGlwIG9mIHRoaXMgY29sbGF0b3IsIAogICAgICogaXQgZG9lcyBub3QgZ2V0IGRlc3Ryb3llZCBkdXJpbmcgdGhpcyBpbnN0YW5jZSdzIGRlc3RydWN0aW9uLgogICAgICogQHBhcmFtIHBhdHRlcm4gVGhlIHRleHQgZm9yIHdoaWNoIHRoaXMgb2JqZWN0IHdpbGwgc2VhcmNoLgogICAgICogQHBhcmFtIHRleHQgICAgVGhlIHRleHQgaW4gd2hpY2ggdG8gc2VhcmNoIGZvciB0aGUgcGF0dGVybi4KICAgICAqIEBwYXJhbSBjb2xsICAgIEEgPHR0PlJ1bGVCYXNlZENvbGxhdG9yPC90dD4gb2JqZWN0IHdoaWNoIGRlZmluZXMgCiAgICAgKiAgICAgICAgICAgICAgICB0aGUgbGFuZ3VhZ2Utc2Vuc2l0aXZlIGNvbXBhcmlzb24gcnVsZXMgdXNlZCB0byAKICAgICAqICAgICAgICAgICAgICAgIGRldGVybWluZSB3aGV0aGVyIHRleHQgaW4gdGhlIHBhdHRlcm4gYW5kIHRhcmdldCAKICAgICAqICAgICAgICAgICAgICAgIG1hdGNoZXMuIFVzZXIgaXMgcmVzcG9uc2libGUgZm9yIHRoZSBjbGVhcmluZyBvZiB0aGlzCiAgICAgKiAgICAgICAgICAgICAgICBvYmplY3QuCiAgICAgKiBAcGFyYW0gYnJlYWtpdGVyIEEgPHR0PkJyZWFrSXRlcmF0b3I8L3R0PiBvYmplY3QgdXNlZCB0byBjb25zdHJhaW4gCiAgICAgKiAgICAgICAgICAgICAgICB0aGUgbWF0Y2hlcyB0aGF0IGFyZSBmb3VuZC4gTWF0Y2hlcyB3aG9zZSBzdGFydCBhbmQgZW5kIAogICAgICogICAgICAgICAgICAgICAgaW5kaWNlcyBpbiB0aGUgdGFyZ2V0IHRleHQgYXJlIG5vdCBib3VuZGFyaWVzIGFzIAogICAgICogICAgICAgICAgICAgICAgZGV0ZXJtaW5lZCBieSB0aGUgPHR0PkJyZWFrSXRlcmF0b3I8L3R0PiBhcmUgCiAgICAgKiAgICAgICAgICAgICAgICBpZ25vcmVkLiBJZiB0aGlzIGJlaGF2aW9yIGlzIG5vdCBkZXNpcmVkLCAKICAgICAqICAgICAgICAgICAgICAgIDx0dD5OVUxMPC90dD4gY2FuIGJlIHBhc3NlZCBpbiBpbnN0ZWFkLgogICAgICogQHBhcmFtIHN0YXR1cyBmb3IgZXJyb3JzIGlmIGFueS4gSWYgZWl0aGVyIHRoZSBsZW5ndGggb2YgcGF0dGVybiBvciAKICAgICAqICAgICAgICAgICAgICAgdGV4dCBpcyAwIHRoZW4gYW4gVV9JTExFR0FMX0FSR1VNRU5UX0VSUk9SIGlzIHJldHVybmVkLgogICAgICogQGRyYWZ0IElDVSAyLjAKICAgICAqLwogICAgU3RyaW5nU2VhcmNoKGNvbnN0IFVuaWNvZGVTdHJpbmcgICAgICZwYXR0ZXJuLCAKICAgICAgICAgICAgICAgICBjb25zdCBVbmljb2RlU3RyaW5nICAgICAmdGV4dCwKICAgICAgICAgICAgICAgICAgICAgICBSdWxlQmFzZWRDb2xsYXRvciAqY29sbCwgICAgICAgCiAgICAgICAgICAgICAgICAgICAgICAgQnJlYWtJdGVyYXRvciAgICAgKmJyZWFraXRlciwKICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICAgICAgICAmc3RhdHVzKTsKCiAgICAvKioKICAgICAqIENyZWF0aW5nIGEgPHR0PlN0cmluZ1NlYXJjaDwvdHQ+IGluc3RhbmNlIHVzaW5nIHRoZSBhcmd1bWVudCBsb2NhbGUgCiAgICAgKiBsYW5ndWFnZSBydWxlIHNldC4gQSBjb2xsYXRvciB3aWxsIGJlIGNyZWF0ZWQgaW4gdGhlIHByb2Nlc3MsIHdoaWNoIAogICAgICogd2lsbCBiZSBvd25lZCBieSB0aGlzIGluc3RhbmNlIGFuZCB3aWxsIGJlIGRlbGV0ZWQgaW4gZHVyaW5nIAogICAgICogZGVzdHJ1Y3Rpb24KICAgICAqIDxwPgogICAgICogTm90ZTogTm8gcGFyc2luZyBvZiB0aGUgdGV4dCB3aXRoaW4gdGhlIDx0dD5DaGFyYWN0ZXJJdGVyYXRvcjwvdHQ+IAogICAgICogd2lsbCBiZSBkb25lIGR1cmluZyBzZWFyY2hpbmcgZm9yIHRoaXMgdmVyc2lvbi4gVGhlIGJsb2NrIG9mIHRleHQgCiAgICAgKiBpbiA8dHQ+Q2hhcmFjdGVySXRlcmF0b3I8L3R0PiB3aWxsIGJlIHVzZWQgYXMgaXQgaXMuCiAgICAgKiBAcGFyYW0gcGF0dGVybiBUaGUgdGV4dCBmb3Igd2hpY2ggdGhpcyBvYmplY3Qgd2lsbCBzZWFyY2guCiAgICAgKiBAcGFyYW0gdGV4dCAgICBUaGUgdGV4dCBpdGVyYXRvciBpbiB3aGljaCB0byBzZWFyY2ggZm9yIHRoZSBwYXR0ZXJuLgogICAgICogQHBhcmFtIGxvY2FsZSAgQSBsb2NhbGUgd2hpY2ggZGVmaW5lcyB0aGUgbGFuZ3VhZ2Utc2Vuc2l0aXZlIAogICAgICogICAgICAgICAgICAgICAgY29tcGFyaXNvbiBydWxlcyB1c2VkIHRvIGRldGVybWluZSB3aGV0aGVyIHRleHQgaW4gdGhlIAogICAgICogICAgICAgICAgICAgICAgcGF0dGVybiBhbmQgdGFyZ2V0IG1hdGNoZXMuIFVzZXIgaXMgcmVzcG9uc2libGUgZm9yIAogICAgICogICAgICAgICAgICAgICAgdGhlIGNsZWFyaW5nIG9mIHRoaXMgb2JqZWN0LgogICAgICogQHBhcmFtIGJyZWFraXRlciBBIDx0dD5CcmVha0l0ZXJhdG9yPC90dD4gb2JqZWN0IHVzZWQgdG8gY29uc3RyYWluIAogICAgICogICAgICAgICAgICAgICAgdGhlIG1hdGNoZXMgdGhhdCBhcmUgZm91bmQuIE1hdGNoZXMgd2hvc2Ugc3RhcnQgYW5kIGVuZCAKICAgICAqICAgICAgICAgICAgICAgIGluZGljZXMgaW4gdGhlIHRhcmdldCB0ZXh0IGFyZSBub3QgYm91bmRhcmllcyBhcyAKICAgICAqICAgICAgICAgICAgICAgIGRldGVybWluZWQgYnkgdGhlIDx0dD5CcmVha0l0ZXJhdG9yPC90dD4gYXJlIAogICAgICogICAgICAgICAgICAgICAgaWdub3JlZC4gSWYgdGhpcyBiZWhhdmlvciBpcyBub3QgZGVzaXJlZCwgCiAgICAgKiAgICAgICAgICAgICAgICA8dHQ+TlVMTDwvdHQ+IGNhbiBiZSBwYXNzZWQgaW4gaW5zdGVhZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgZm9yIGVycm9ycyBpZiBhbnkuIElmIGVpdGhlciB0aGUgbGVuZ3RoIG9mIHBhdHRlcm4gb3IgCiAgICAgKiAgICAgICAgICAgICAgIHRleHQgaXMgMCB0aGVuIGFuIFVfSUxMRUdBTF9BUkdVTUVOVF9FUlJPUiBpcyByZXR1cm5lZC4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIFN0cmluZ1NlYXJjaChjb25zdCBVbmljb2RlU3RyaW5nICZwYXR0ZXJuLCBDaGFyYWN0ZXJJdGVyYXRvciAmdGV4dCwKICAgICAgICAgICAgICAgICBjb25zdCBMb2NhbGUgICAgICAgICZsb2NhbGUsIAogICAgICAgICAgICAgICAgICAgICAgIEJyZWFrSXRlcmF0b3IgKmJyZWFraXRlciwKICAgICAgICAgICAgICAgICAgICAgICBVRXJyb3JDb2RlICAgICZzdGF0dXMpOwoKICAgIC8qKgogICAgICogQ3JlYXRpbmcgYSA8dHQ+U3RyaW5nU2VhcmNoPC90dD4gaW5zdGFuY2UgdXNpbmcgdGhlIGFyZ3VtZW50IGNvbGxhdG9yIAogICAgICogbGFuZ3VhZ2UgcnVsZSBzZXQuIE5vdGUsIHVzZXIgcmV0YWlucyB0aGUgb3duZXJzaGlwIG9mIHRoaXMgY29sbGF0b3IsIAogICAgICogaXQgZG9lcyBub3QgZ2V0IGRlc3Ryb3llZCBkdXJpbmcgdGhpcyBpbnN0YW5jZSdzIGRlc3RydWN0aW9uLgogICAgICogPHA+CiAgICAgKiBOb3RlOiBObyBwYXJzaW5nIG9mIHRoZSB0ZXh0IHdpdGhpbiB0aGUgPHR0PkNoYXJhY3Rlckl0ZXJhdG9yPC90dD4gCiAgICAgKiB3aWxsIGJlIGRvbmUgZHVyaW5nIHNlYXJjaGluZyBmb3IgdGhpcyB2ZXJzaW9uLiBUaGUgYmxvY2sgb2YgdGV4dCAKICAgICAqIGluIDx0dD5DaGFyYWN0ZXJJdGVyYXRvcjwvdHQ+IHdpbGwgYmUgdXNlZCBhcyBpdCBpcy4KICAgICAqIEBwYXJhbSBwYXR0ZXJuIFRoZSB0ZXh0IGZvciB3aGljaCB0aGlzIG9iamVjdCB3aWxsIHNlYXJjaC4KICAgICAqIEBwYXJhbSB0ZXh0ICAgIFRoZSB0ZXh0IGluIHdoaWNoIHRvIHNlYXJjaCBmb3IgdGhlIHBhdHRlcm4uCiAgICAgKiBAcGFyYW0gY29sbCAgICBBIDx0dD5SdWxlQmFzZWRDb2xsYXRvcjwvdHQ+IG9iamVjdCB3aGljaCBkZWZpbmVzIAogICAgICogICAgICAgICAgICAgICAgdGhlIGxhbmd1YWdlLXNlbnNpdGl2ZSBjb21wYXJpc29uIHJ1bGVzIHVzZWQgdG8gCiAgICAgKiAgICAgICAgICAgICAgICBkZXRlcm1pbmUgd2hldGhlciB0ZXh0IGluIHRoZSBwYXR0ZXJuIGFuZCB0YXJnZXQgCiAgICAgKiAgICAgICAgICAgICAgICBtYXRjaGVzLiBVc2VyIGlzIHJlc3BvbnNpYmxlIGZvciB0aGUgY2xlYXJpbmcgb2YgdGhpcwogICAgICogICAgICAgICAgICAgICAgb2JqZWN0LgogICAgICogQHBhcmFtIGJyZWFraXRlciBBIDx0dD5CcmVha0l0ZXJhdG9yPC90dD4gb2JqZWN0IHVzZWQgdG8gY29uc3RyYWluIAogICAgICogICAgICAgICAgICAgICAgdGhlIG1hdGNoZXMgdGhhdCBhcmUgZm91bmQuIE1hdGNoZXMgd2hvc2Ugc3RhcnQgYW5kIGVuZCAKICAgICAqICAgICAgICAgICAgICAgIGluZGljZXMgaW4gdGhlIHRhcmdldCB0ZXh0IGFyZSBub3QgYm91bmRhcmllcyBhcyAKICAgICAqICAgICAgICAgICAgICAgIGRldGVybWluZWQgYnkgdGhlIDx0dD5CcmVha0l0ZXJhdG9yPC90dD4gYXJlIAogICAgICogICAgICAgICAgICAgICAgaWdub3JlZC4gSWYgdGhpcyBiZWhhdmlvciBpcyBub3QgZGVzaXJlZCwgCiAgICAgKiAgICAgICAgICAgICAgICA8dHQ+TlVMTDwvdHQ+IGNhbiBiZSBwYXNzZWQgaW4gaW5zdGVhZC4KICAgICAqIEBwYXJhbSBzdGF0dXMgZm9yIGVycm9ycyBpZiBhbnkuIElmIGVpdGhlciB0aGUgbGVuZ3RoIG9mIHBhdHRlcm4gb3IgCiAgICAgKiAgICAgICAgICAgICAgIHRleHQgaXMgMCB0aGVuIGFuIFVfSUxMRUdBTF9BUkdVTUVOVF9FUlJPUiBpcyByZXR1cm5lZC4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIFN0cmluZ1NlYXJjaChjb25zdCBVbmljb2RlU3RyaW5nICAgICAmcGF0dGVybiwgQ2hhcmFjdGVySXRlcmF0b3IgJnRleHQsCiAgICAgICAgICAgICAgICAgICAgICAgUnVsZUJhc2VkQ29sbGF0b3IgKmNvbGwsIAogICAgICAgICAgICAgICAgICAgICAgIEJyZWFrSXRlcmF0b3IgICAgICpicmVha2l0ZXIsCiAgICAgICAgICAgICAgICAgICAgICAgVUVycm9yQ29kZSAgICAgICAgJnN0YXR1cyk7CgogICAgLyoqCiAgICAgKiBDb3B5IGNvbnN0cnVjdG9yIHRoYXQgY3JlYXRlcyBhIFN0cmluZ1NlYXJjaCBpbnN0YW5jZSB3aXRoIHRoZSBzYW1lIAogICAgICogYmVoYXZpb3IsIGFuZCBpdGVyYXRpbmcgb3ZlciB0aGUgc2FtZSB0ZXh0LgogICAgICogQHBhcmFtIHRoYXQgU3RyaW5nU2VhcmNoIGluc3RhbmNlIHRvIGJlIGNvcGllZC4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIFN0cmluZ1NlYXJjaChjb25zdCBTdHJpbmdTZWFyY2ggJnRoYXQpOwoKICAgIC8qKgogICAgKiBEZXN0cnVjdG9yLiBDbGVhbnMgdXAgdGhlIHNlYXJjaCBpdGVyYXRvciBkYXRhIHN0cnVjdC4KICAgICogSWYgYSBjb2xsYXRvciBpcyBjcmVhdGVkIGluIHRoZSBjb25zdHJ1Y3RvciwgaXQgd2lsbCBiZSBkZXN0cm95ZWQgaGVyZS4KICAgICogQGRyYWZ0IElDVSAyLjAKICAgICovCiAgICB2aXJ0dWFsIH5TdHJpbmdTZWFyY2godm9pZCk7CgogICAgLy8gb3BlcmF0b3Igb3ZlcmxvYWRpbmcgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgogICAgLyoqCiAgICAgKiBBc3NpZ25tZW50IG9wZXJhdG9yLiBTZXRzIHRoaXMgaXRlcmF0b3IgdG8gaGF2ZSB0aGUgc2FtZSBiZWhhdmlvciwKICAgICAqIGFuZCBpdGVyYXRlIG92ZXIgdGhlIHNhbWUgdGV4dCwgYXMgdGhlIG9uZSBwYXNzZWQgaW4uCiAgICAgKiBAcGFyYW0gdGhhdCBpbnN0YW5jZSB0byBiZSBjb3BpZWQuCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICBTdHJpbmdTZWFyY2ggJiBvcGVyYXRvcj0oY29uc3QgU3RyaW5nU2VhcmNoICZ0aGF0KTsKCiAgICAvKioKICAgICAqIEVxdWFsaXR5IG9wZXJhdG9yLiAKICAgICAqIEBwYXJhbSB0aGF0IGluc3RhbmNlIHRvIGJlIGNvbXBhcmVkLgogICAgICogQHJldHVybiBUUlVFIGlmIGJvdGggaW5zdGFuY2VzIGhhdmUgdGhlIHNhbWUgYXR0cmlidXRlcywgCiAgICAgKiAgICAgICAgIGJyZWFraXRlcmF0b3JzLCBjb2xsYXRvcnMgYW5kIGl0ZXJhdGUgb3ZlciB0aGUgc2FtZSB0ZXh0IAogICAgICogICAgICAgICB3aGlsZSBsb29raW5nIGZvciB0aGUgc2FtZSBwYXR0ZXJuLgogICAgICogQGRyYWZ0IElDVSAyLjAKICAgICAqLwogICAgdmlydHVhbCBVQm9vbCBvcGVyYXRvcj09KGNvbnN0IFNlYXJjaEl0ZXJhdG9yICZ0aGF0KSBjb25zdDsKCiAgICAvLyBwdWJsaWMgZ2V0IGFuZCBzZXQgbWV0aG9kcyAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgogICAgLyoqCiAgICAgKiBTZXRzIHRoZSBpbmRleCB0byBwb2ludCB0byB0aGUgZ2l2ZW4gcG9zaXRpb24sIGFuZCBjbGVhcnMgYW55IHN0YXRlIAogICAgICogdGhhdCdzIGFmZmVjdGVkLgogICAgICogPHA+CiAgICAgKiBUaGlzIG1ldGhvZCB0YWtlcyB0aGUgYXJndW1lbnQgaW5kZXggYW5kIHNldHMgdGhlIHBvc2l0aW9uIGluIHRoZSB0ZXh0IAogICAgICogc3RyaW5nIGFjY29yZGluZ2x5IHdpdGhvdXQgY2hlY2tpbmcgaWYgdGhlIGluZGV4IGlzIHBvaW50aW5nIHRvIGEgCiAgICAgKiB2YWxpZCBzdGFydGluZyBwb2ludCB0byBiZWdpbiBzZWFyY2hpbmcuIAogICAgICogQHBhcmFtIHBvc2l0aW9uIHdpdGhpbiB0aGUgdGV4dCB0byBiZSBzZXQuIElmIHBvc2l0aW9uIGlzIGxlc3MKICAgICAqIAkJCXRoYW4gb3IgZ3JlYXRlciB0aGFuIHRoZSB0ZXh0IHJhbmdlIGZvciBzZWFyY2hpbmcsIAogICAgICogICAgICAgICAgYW4gVV9JTkRFWF9PVVRPRkJPVU5EU19FUlJPUiB3aWxsIGJlIHJldHVybmVkCiAgICAgKiBAcGFyYW0gc3RhdHVzIGZvciBlcnJvcnMgaWYgaXQgb2NjdXJzCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICB2aXJ0dWFsIHZvaWQgc2V0T2Zmc2V0KGludDMyX3QgcG9zaXRpb24sIFVFcnJvckNvZGUgJnN0YXR1cyk7CgogICAgLyoqCiAgICAgKiBSZXR1cm4gdGhlIGN1cnJlbnQgaW5kZXggaW4gdGhlIHRleHQgYmVpbmcgc2VhcmNoZWQuCiAgICAgKiBJZiB0aGUgaXRlcmF0aW9uIGhhcyBnb25lIHBhc3QgdGhlIGVuZCBvZiB0aGUgdGV4dAogICAgICogKG9yIHBhc3QgdGhlIGJlZ2lubmluZyBmb3IgYSBiYWNrd2FyZHMgc2VhcmNoKSwge0BsaW5rICNVU0VBUkNIX0RPTkV9IAogICAgICogaXMgcmV0dXJuZWQuCiAgICAgKiBAcmV0dXJuIGN1cnJlbnQgaW5kZXggaW4gdGhlIHRleHQgYmVpbmcgc2VhcmNoZWQuCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICB2aXJ0dWFsIGludDMyX3QgZ2V0T2Zmc2V0KHZvaWQpIGNvbnN0OwoKICAgIC8qKgogICAgICogU2V0IHRoZSB0YXJnZXQgdGV4dCB0byBiZSBzZWFyY2hlZC4KICAgICAqIFRleHQgaXRlcmF0aW9uIHdpbGwgaGVuY2UgYmVnaW4gYXQgdGhlIHN0YXJ0IG9mIHRoZSB0ZXh0IHN0cmluZy4gCiAgICAgKiBUaGlzIG1ldGhvZCBpcyAKICAgICAqIHVzZWZ1bCBpZiB5b3Ugd2FudCB0byByZS11c2UgYW4gaXRlcmF0b3IgdG8gc2VhcmNoIGZvciB0aGUgc2FtZSAKICAgICAqIHBhdHRlcm4gd2l0aGluIGEgZGlmZmVyZW50IGJvZHkgb2YgdGV4dC4KICAgICAqIEBwYXJhbSB0ZXh0IHRleHQgc3RyaW5nIHRvIGJlIHNlYXJjaGVkCiAgICAgKiBAcGFyYW0gc3RhdHVzIGZvciBlcnJvcnMgaWYgYW55LiBJZiB0aGUgdGV4dCBsZW5ndGggaXMgMCB0aGVuIGFuIAogICAgICogICAgICAgIFVfSUxMRUdBTF9BUkdVTUVOVF9FUlJPUiBpcyByZXR1cm5lZC4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCBzZXRUZXh0KGNvbnN0IFVuaWNvZGVTdHJpbmcgJnRleHQsIFVFcnJvckNvZGUgJnN0YXR1cyk7CiAgICAKICAgIC8qKgogICAgICogU2V0IHRoZSB0YXJnZXQgdGV4dCB0byBiZSBzZWFyY2hlZC4KICAgICAqIFRleHQgaXRlcmF0aW9uIHdpbGwgaGVuY2UgYmVnaW4gYXQgdGhlIHN0YXJ0IG9mIHRoZSB0ZXh0IHN0cmluZy4gCiAgICAgKiBUaGlzIG1ldGhvZCBpcyAKICAgICAqIHVzZWZ1bCBpZiB5b3Ugd2FudCB0byByZS11c2UgYW4gaXRlcmF0b3IgdG8gc2VhcmNoIGZvciB0aGUgc2FtZSAKICAgICAqIHBhdHRlcm4gd2l0aGluIGEgZGlmZmVyZW50IGJvZHkgb2YgdGV4dC4KICAgICAqIE5vdGU6IE5vIHBhcnNpbmcgb2YgdGhlIHRleHQgd2l0aGluIHRoZSA8dHQ+Q2hhcmFjdGVySXRlcmF0b3I8L3R0PiAKICAgICAqIHdpbGwgYmUgZG9uZSBkdXJpbmcgc2VhcmNoaW5nIGZvciB0aGlzIHZlcnNpb24uIFRoZSBibG9jayBvZiB0ZXh0IAogICAgICogaW4gPHR0PkNoYXJhY3Rlckl0ZXJhdG9yPC90dD4gd2lsbCBiZSB1c2VkIGFzIGl0IGlzLgogICAgICogQHBhcmFtIHRleHQgdGV4dCBzdHJpbmcgdG8gYmUgc2VhcmNoZWQKICAgICAqIEBwYXJhbSBzdGF0dXMgZm9yIGVycm9ycyBpZiBhbnkuIElmIHRoZSB0ZXh0IGxlbmd0aCBpcyAwIHRoZW4gYW4gCiAgICAgKiAgICAgICAgVV9JTExFR0FMX0FSR1VNRU5UX0VSUk9SIGlzIHJldHVybmVkLgogICAgICogQGRyYWZ0IElDVSAyLjAKICAgICAqLwogICAgdmlydHVhbCB2b2lkIHNldFRleHQoQ2hhcmFjdGVySXRlcmF0b3IgJnRleHQsIFVFcnJvckNvZGUgJnN0YXR1cyk7CgogICAgLyoqCiAgICAgKiBHZXRzIHRoZSBjb2xsYXRvciB1c2VkIGZvciB0aGUgbGFuZ3VhZ2UgcnVsZXMuIAogICAgICogPHA+CiAgICAgKiBEZWxldGluZyB0aGUgcmV0dXJuZWQgPHR0PlJ1bGVCYXNlZENvbGxhdG9yPC90dD4gYmVmb3JlIGNhbGxpbmcgCiAgICAgKiB0aGUgZGVzdHJ1Y3RvciB3b3VsZCBjYXVzZSB0aGUgc3RyaW5nIHNlYXJjaCB0byBmYWlsLgogICAgICogVGhlIGRlc3RydWN0b3Igd2lsbCBkZWxldGUgdGhlIGNvbGxhdG9yIGlmIHRoaXMgaW5zdGFuY2Ugb3ducyBpdAogICAgICogQHJldHVybiBjb2xsYXRvciB1c2VkIGZvciBzdHJpbmcgc2VhcmNoCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICBSdWxlQmFzZWRDb2xsYXRvciAqIGdldENvbGxhdG9yKCkgY29uc3Q7CiAgICAKICAgIC8qKgogICAgICogU2V0cyB0aGUgY29sbGF0b3IgdXNlZCBmb3IgdGhlIGxhbmd1YWdlIHJ1bGVzLiBVc2VyIHJldGFpbnMgdGhlIAogICAgICogb3duZXJzaGlwIG9mIHRoaXMgY29sbGF0b3IsIHRodXMgdGhlIHJlc3BvbnNpYmlsaXR5IG9mIGRlbGV0aW9uIGxpZXMgCiAgICAgKiB3aXRoIHRoZSB1c2VyLiBUaGlzIG1ldGhvZCBjYXVzZXMgaW50ZXJuYWwgZGF0YSBzdWNoIGFzIEJveWVyLU1vb3JlIAogICAgICogc2hpZnQgdGFibGVzIHRvIGJlIHJlY2FsY3VsYXRlZCwgYnV0IHRoZSBpdGVyYXRvcidzIHBvc2l0aW9uIGlzIAogICAgICogdW5jaGFuZ2VkLgogICAgICogQHBhcmFtIGNvbGwgICAgY29sbGF0b3IgCiAgICAgKiBAcGFyYW0gc3RhdHVzICBmb3IgZXJyb3JzIGlmIGFueQogICAgICogQGRyYWZ0IElDVSAyLjAKICAgICAqLwogICAgdm9pZCBzZXRDb2xsYXRvcihSdWxlQmFzZWRDb2xsYXRvciAqY29sbCwgVUVycm9yQ29kZSAmc3RhdHVzKTsKICAgIAogICAgLyoqCiAgICAgKiBTZXRzIHRoZSBwYXR0ZXJuIHVzZWQgZm9yIG1hdGNoaW5nLgogICAgICogSW50ZXJuYWwgZGF0YSBsaWtlIHRoZSBCb3llciBNb29yZSB0YWJsZSB3aWxsIGJlIHJlY2FsY3VsYXRlZCwgYnV0IAogICAgICogdGhlIGl0ZXJhdG9yJ3MgcG9zaXRpb24gaXMgdW5jaGFuZ2VkLgogICAgICogQHBhcmFtIHBhdHRlcm4gc2VhcmNoIHBhdHRlcm4gdG8gYmUgZm91bmQKICAgICAqIEBwYXJhbSBzdGF0dXMgZm9yIGVycm9ycyBpZiBhbnkuIElmIHRoZSBwYXR0ZXJuIGxlbmd0aCBpcyAwIHRoZW4gYW4gCiAgICAgKiAgICAgICAgICAgICAgIFVfSUxMRUdBTF9BUkdVTUVOVF9FUlJPUiBpcyByZXR1cm5lZC4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIHZvaWQgc2V0UGF0dGVybihjb25zdCBVbmljb2RlU3RyaW5nICZwYXR0ZXJuLCBVRXJyb3JDb2RlICZzdGF0dXMpOwogICAgCiAgICAvKioKICAgICAqIEdldHMgdGhlIHNlYXJjaCBwYXR0ZXJuLgogICAgICogQHJldHVybiBwYXR0ZXJuIHVzZWQgZm9yIG1hdGNoaW5nCiAgICAgKiBAZHJhZnQgSUNVIDIuMAogICAgICovCiAgICBjb25zdCBVbmljb2RlU3RyaW5nICYgZ2V0UGF0dGVybigpIGNvbnN0OwoKICAgIC8vIHB1YmxpYyBtZXRob2RzIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiAgICAvKiogCiAgICAgKiBSZXNldCB0aGUgaXRlcmF0aW9uLgogICAgICogU2VhcmNoIHdpbGwgYmVnaW4gYXQgdGhlIHN0YXJ0IG9mIHRoZSB0ZXh0IHN0cmluZyBpZiBhIGZvcndhcmQgCiAgICAgKiBpdGVyYXRpb24gaXMgaW5pdGlhdGVkIGJlZm9yZSBhIGJhY2t3YXJkcyBpdGVyYXRpb24uIE90aGVyd2lzZSBpZiAKICAgICAqIGEgYmFja3dhcmRzIGl0ZXJhdGlvbiBpcyBpbml0aWF0ZWQgYmVmb3JlIGEgZm9yd2FyZHMgaXRlcmF0aW9uLCB0aGUgCiAgICAgKiBzZWFyY2ggd2lsbCBiZWdpbiBhdCB0aGUgZW5kIG9mIHRoZSB0ZXh0IHN0cmluZy4KICAgICAqIEBkcmFmdCBJQ1UgMi4wCiAgICAgKi8KICAgIHZpcnR1YWwgdm9pZCByZXNldCgpOwoKICAgIC8qKgogICAgICogUmV0dXJucyBhIGNvcHkgb2YgU3RyaW5nU2VhcmNoIHdpdGggdGhlIHNhbWUgYmVoYXZpb3IsIGFuZCAKICAgICAqIGl0ZXJhdGluZyBvdmVyIHRoZSBzYW1lIHRleHQsIGFzIHRoaXMgb25lLiBOb3RlIHRoYXQgYWxsIGRhdGEgd2lsbCBiZQogICAgICogcmVwbGljYXRlZCwgZXhjZXB0IGZvciB0aGUgdXNlci1zcGVjaWZpZWQgY29sbGF0b3IgYW5kIHRoZQogICAgICogYnJlYWtpdGVyYXRvci4KICAgICAqIEByZXR1cm4gY2xvbmVkIG9iamVjdAogICAgICogQGRyYWZ0IElDVSAyLjAKICAgICAqLwogICAgdmlydHVhbCBTZWFyY2hJdGVyYXRvciAqIHNhZmVDbG9uZSh2b2lkKSBjb25zdDsKICAgIAogICAgLyoqCiAgICAgKiBJQ1UgInBvb3IgbWFuJ3MgUlRUSSIsIHJldHVybnMgYSBVQ2xhc3NJRCBmb3IgdGhlIGFjdHVhbCBjbGFzcy4KICAgICAqCiAgICAgKiBAZHJhZnQgSUNVIDIuMgogICAgICovCiAgICB2aXJ0dWFsIGlubGluZSBVQ2xhc3NJRCBnZXREeW5hbWljQ2xhc3NJRCgpIGNvbnN0IHsgcmV0dXJuIGdldFN0YXRpY0NsYXNzSUQoKTsgfQoKICAgIC8qKgogICAgICogSUNVICJwb29yIG1hbidzIFJUVEkiLCByZXR1cm5zIGEgVUNsYXNzSUQgZm9yIHRoaXMgY2xhc3MuCiAgICAgKgogICAgICogQGRyYWZ0IElDVSAyLjIKICAgICAqLwogICAgc3RhdGljIGlubGluZSBVQ2xhc3NJRCBnZXRTdGF0aWNDbGFzc0lEKCkgeyByZXR1cm4gKFVDbGFzc0lEKSZmZ0NsYXNzSUQ7IH0KCnByb3RlY3RlZDoKCiAgICAvLyBwcm90ZWN0ZWQgbWV0aG9kIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiAgICAvKioKICAgICAqIFNlYXJjaCBmb3J3YXJkIGZvciBtYXRjaGluZyB0ZXh0LCBzdGFydGluZyBhdCBhIGdpdmVuIGxvY2F0aW9uLgogICAgICogQ2xpZW50cyBzaG91bGQgbm90IGNhbGwgdGhpcyBtZXRob2QgZGlyZWN0bHk7IGluc3RlYWQgdGhleSBzaG91bGQgCiAgICAgKiBjYWxsIHtAbGluayBTZWFyY2hJdGVyYXRvciNuZXh0fS4KICAgICAqIDxwPgogICAgICogSWYgYSBtYXRjaCBpcyBmb3VuZCwgdGhpcyBtZXRob2QgcmV0dXJucyB0aGUgaW5kZXggYXQgd2hpY2ggdGhlIG1hdGNoCiAgICAgKiBzdGFydHMgYW5kIGNhbGxzIHtAbGluayBTZWFyY2hJdGVyYXRvciNzZXRNYXRjaExlbmd0aH0gd2l0aCB0aGUgbnVtYmVyIAogICAgICogb2YgY2hhcmFjdGVycyBpbiB0aGUgdGFyZ2V0IHRleHQgdGhhdCBtYWtlIHVwIHRoZSBtYXRjaC4gSWYgbm8gbWF0Y2ggCiAgICAgKiBpcyBmb3VuZCwgdGhlIG1ldGhvZCByZXR1cm5zIDx0dD5VU0VBUkNIX0RPTkU8L3R0Pi4KICAgICAqIDxwPgogICAgICogVGhlIDx0dD5TdHJpbmdTZWFyY2g8L3R0PiBpcyBhZGp1c3RlZCBzbyB0aGF0IGl0cyBjdXJyZW50IGluZGV4IAogICAgICogKGFzIHJldHVybmVkIGJ5IHtAbGluayAjZ2V0T2Zmc2V0fSkgaXMgdGhlIG1hdGNoIHBvc2l0aW9uIGlmIG9uZSB3YXMgCiAgICAgKiBmb3VuZC4KICAgICAqIElmIGEgbWF0Y2ggaXMgbm90IGZvdW5kLCA8dHQ+VVNFQVJDSF9ET05FPC90dD4gd2lsbCBiZSByZXR1cm5lZCBhbmQKICAgICAqIHRoZSA8dHQ+U3RyaW5nU2VhcmNoPC90dD4gd2lsbCBiZSBhZGp1c3RlZCB0byB0aGUgaW5kZXggVVNFQVJDSF9ET05FLgogICAgICogQHBhcmFtIHBvc2l0aW9uIFRoZSBpbmRleCBpbiB0aGUgdGFyZ2V0IHRleHQgYXQgd2hpY2ggdGhlIHNlYXJjaCAKICAgICAqICAgICAgICAgICAgICAgICBzdGFydHMKICAgICAqIEBwYXJhbSBzdGF0dXMgZm9yIGVycm9ycyBpZiBhbnkgb2NjdXJzCiAgICAgKiBAcmV0dXJuIFRoZSBpbmRleCBhdCB3aGljaCB0aGUgbWF0Y2hlZCB0ZXh0IGluIHRoZSB0YXJnZXQgc3RhcnRzLCBvciAKICAgICAqICAgICAgICAgVVNFQVJDSF9ET05FIGlmIG5vIG1hdGNoIHdhcyBmb3VuZC4KICAgICAqLwogICAgdmlydHVhbCBpbnQzMl90IGhhbmRsZU5leHQoaW50MzJfdCBwb3NpdGlvbiwgVUVycm9yQ29kZSAmc3RhdHVzKTsKCiAgICAvKioKICAgICAqIFNlYXJjaCBiYWNrd2FyZCBmb3IgbWF0Y2hpbmcgdGV4dCwgc3RhcnRpbmcgYXQgYSBnaXZlbiBsb2NhdGlvbi4KICAgICAqIENsaWVudHMgc2hvdWxkIG5vdCBjYWxsIHRoaXMgbWV0aG9kIGRpcmVjdGx5OyBpbnN0ZWFkIHRoZXkgc2hvdWxkIGNhbGwKICAgICAqIDx0dD5TZWFyY2hJdGVyYXRvci5wcmV2aW91cygpPC90dD4sIHdoaWNoIHRoaXMgbWV0aG9kIG92ZXJyaWRlcy4KICAgICAqIDxwPgogICAgICogSWYgYSBtYXRjaCBpcyBmb3VuZCwgdGhpcyBtZXRob2QgcmV0dXJucyB0aGUgaW5kZXggYXQgd2hpY2ggdGhlIG1hdGNoCiAgICAgKiBzdGFydHMgYW5kIGNhbGxzIHtAbGluayBTZWFyY2hJdGVyYXRvciNzZXRNYXRjaExlbmd0aH0gd2l0aCB0aGUgbnVtYmVyIAogICAgICogb2YgY2hhcmFjdGVycyBpbiB0aGUgdGFyZ2V0IHRleHQgdGhhdCBtYWtlIHVwIHRoZSBtYXRjaC4gSWYgbm8gbWF0Y2ggCiAgICAgKiBpcyBmb3VuZCwgdGhlIG1ldGhvZCByZXR1cm5zIDx0dD5VU0VBUkNIX0RPTkU8L3R0Pi4KICAgICAqIDxwPgogICAgICogVGhlIDx0dD5TdHJpbmdTZWFyY2g8L3R0PiBpcyBhZGp1c3RlZCBzbyB0aGF0IGl0cyBjdXJyZW50IGluZGV4IAogICAgICogKGFzIHJldHVybmVkIGJ5IHtAbGluayAjZ2V0T2Zmc2V0fSkgaXMgdGhlIG1hdGNoIHBvc2l0aW9uIGlmIG9uZSB3YXMgCiAgICAgKiBmb3VuZC4KICAgICAqIElmIGEgbWF0Y2ggaXMgbm90IGZvdW5kLCA8dHQ+VVNFQVJDSF9ET05FPC90dD4gd2lsbCBiZSByZXR1cm5lZCBhbmQKICAgICAqIHRoZSA8dHQ+U3RyaW5nU2VhcmNoPC90dD4gd2lsbCBiZSBhZGp1c3RlZCB0byB0aGUgaW5kZXggVVNFQVJDSF9ET05FLgogICAgICogQHBhcmFtIHBvc2l0aW9uIFRoZSBpbmRleCBpbiB0aGUgdGFyZ2V0IHRleHQgYXQgd2hpY2ggdGhlIHNlYXJjaCAKICAgICAqICAgICAgICAgICAgICAgICBzdGFydHMuCiAgICAgKiBAcGFyYW0gc3RhdHVzIGZvciBlcnJvcnMgaWYgYW55IG9jY3VycwogICAgICogQHJldHVybiBUaGUgaW5kZXggYXQgd2hpY2ggdGhlIG1hdGNoZWQgdGV4dCBpbiB0aGUgdGFyZ2V0IHN0YXJ0cywgb3IgCiAgICAgKiAgICAgICAgIFVTRUFSQ0hfRE9ORSBpZiBubyBtYXRjaCB3YXMgZm91bmQuCiAgICAgKi8KICAgIHZpcnR1YWwgaW50MzJfdCBoYW5kbGVQcmV2KGludDMyX3QgcG9zaXRpb24sIFVFcnJvckNvZGUgJnN0YXR1cyk7CiAgICAKcHJpdmF0ZSA6CgogICAgLy8gcHJpdmF0ZSBkYXRhIG1lbWJlcnMgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKICAgIC8qKgogICAgKiBSdWxlQmFzZWRDb2xsYXRvciwgY29udGFpbnMgZXhhY3RseSB0aGUgc2FtZSBVQ29sbGF0b3IgKiBpbiBtX3N0cnNyY2hfCiAgICAqLwogICAgUnVsZUJhc2VkQ29sbGF0b3IgIG1fY29sbGF0b3JfOwogICAgLyoqCiAgICAqIFBhdHRlcm4gdGV4dAogICAgKi8KICAgIFVuaWNvZGVTdHJpbmcgICAgICBtX3BhdHRlcm5fOwogICAgLyoqCiAgICAqIENvcnJlc3BvbmRpbmcgY29sbGF0aW9uIHJ1bGVzCiAgICAqLwogICAgVW5pY29kZVN0cmluZyAgICAgIG1fY29sbGF0aW9uX3J1bGVzXzsKICAgIC8qKgogICAgKiBTdHJpbmcgc2VhcmNoIHN0cnVjdCBkYXRhCiAgICAqLwogICAgVVN0cmluZ1NlYXJjaCAgICAgKm1fc3Ryc3JjaF87CgogICAgLyoqCiAgICAgKiBUaGUgYWRkcmVzcyBvZiB0aGlzIHN0YXRpYyBjbGFzcyB2YXJpYWJsZSBzZXJ2ZXMgYXMgdGhpcyBjbGFzcydzIElECiAgICAgKiBmb3IgSUNVICJwb29yIG1hbidzIFJUVEkiLgogICAgICovCiAgICBzdGF0aWMgY29uc3QgY2hhciBmZ0NsYXNzSUQ7Cn07CgpVX05BTUVTUEFDRV9FTkQKCiNlbmRpZgoK